十、JavaScript 工廠模式

工廠模式

工廠模式是另外一種關(guān)注對(duì)象創(chuàng)建概念的創(chuàng)建模式嘿般。它的領(lǐng)域中同其它模式的不同之處在于它并沒(méi)有明確要求我們使用一個(gè)構(gòu)造器。取而代之涯冠,一個(gè)工廠能提供一個(gè)創(chuàng)建對(duì)象的公共接口炉奴,我們可以在其中指定我們希望被創(chuàng)建的工廠對(duì)象的類(lèi)型。

試想一下蛇更,在我們被要求創(chuàng)建一種類(lèi)型的UI組件時(shí)瞻赶,我們就有一個(gè)UI工廠。并不是通過(guò)直接使用new操作符或者通過(guò)另外一個(gè)構(gòu)造器來(lái)創(chuàng)建這個(gè)組件派任,我們?nèi)《南蛞粋€(gè)工廠對(duì)象索要一個(gè)新的組件砸逊。我們告知工廠我們需要什么類(lèi)型的組件(例如:“按鈕”,“面板”)掌逛,而它會(huì)將其初始化师逸,然后返回供我們使用。

如果創(chuàng)建過(guò)程相當(dāng)復(fù)雜的話豆混,那這會(huì)特別的有用篓像,例如:如果它強(qiáng)烈依賴于動(dòng)態(tài)因素或者應(yīng)用程序配置的話。

這個(gè)模式的一些例子可以在UI庫(kù)里面找到崖叫,例如ExtJS, 用于創(chuàng)建對(duì)象或者組件的方法可以被做更深層次的子類(lèi)遗淳。 下面使用用我們之前的那些代碼來(lái)做的一個(gè)例子,通過(guò)使用構(gòu)造器模式邏輯來(lái)定義汽車(chē)心傀。這個(gè)例子展示了Vehicle 工廠可以使用工廠模式來(lái)實(shí)現(xiàn)。

// Types.js - Constructors used behind the scenes

// A constructor for defining new cars
function Car( options ) {

  // some defaults
  this.doors = options.doors || 4;
  this.state = options.state || "brand new";
  this.color = options.color || "silver";

}

// A constructor for defining new trucks
function Truck( options){

  this.state = options.state || "used";
  this.wheelSize = options.wheelSize || "large";
  this.color = options.color || "blue";
}

// FactoryExample.js

// Define a skeleton vehicle factory
function VehicleFactory() {}

// Define the prototypes and utilities for this factory

// Our default vehicleClass is Car
VehicleFactory.prototype.vehicleClass = Car;

// Our Factory method for creating new Vehicle instances
VehicleFactory.prototype.createVehicle = function ( options ) {

  if( options.vehicleType === "car" ){
    this.vehicleClass = Car;
  }else{
    this.vehicleClass = Truck;
  }

  return new this.vehicleClass( options );

};

// Create an instance of our factory that makes cars
var carFactory = new VehicleFactory();
var car = carFactory.createVehicle( {
            vehicleType: "car",
            color: "yellow",
            doors: 6 } );

// Test to confirm our car was created using the vehicleClass/prototype Car

// Outputs: true
console.log( car instanceof Car );

// Outputs: Car object of color "yellow", doors: 6 in a "brand new" state
console.log( car );

方法1: 修改 VehicleFactory 實(shí)例使用 Truck 類(lèi)

var movingTruck = carFactory.createVehicle( {
                      vehicleType: "truck",
                      state: "like new",
                      color: "red",
                      wheelSize: "small" } );

// Test to confirm our truck was created with the vehicleClass/prototype Truck

// Outputs: true
console.log( movingTruck instanceof Truck );

// Outputs: Truck object of color "red", a "like new" state
// and a "small" wheelSize
console.log( movingTruck );

方法2: 做 VehicleFactory 的子類(lèi)用于創(chuàng)建一個(gè)工廠類(lèi)生產(chǎn) Trucks

function TruckFactory () {}
TruckFactory.prototype = new VehicleFactory();
TruckFactory.prototype.vehicleClass = Truck;

var truckFactory = new TruckFactory();
var myBigTruck = truckFactory.createVehicle( {
                    state: "omg..so bad.",
                    color: "pink",
                    wheelSize: "so big" } );

// Confirms that myBigTruck was created with the prototype Truck
// Outputs: true
console.log( myBigTruck instanceof Truck );

// Outputs: Truck object with the color "pink", wheelSize "so big"
// and state "omg. so bad"
console.log( myBigTruck );
何時(shí)使用工廠模式

當(dāng)被應(yīng)用到下面的場(chǎng)景中時(shí)拆讯,工廠模式特別有用:

當(dāng)我們的對(duì)象或者組件設(shè)置涉及到高程度級(jí)別的復(fù)雜度時(shí)脂男。
當(dāng)我們需要根據(jù)我們所在的環(huán)境方便的生成不同對(duì)象的實(shí)體時(shí)养叛。
當(dāng)我們?cè)谠S多共享同一個(gè)屬性的許多小型對(duì)象或組件上工作時(shí)。
當(dāng)帶有其它僅僅需要滿足一種API約定(又名鴨式類(lèi)型)的對(duì)象的組合對(duì)象工作時(shí).這對(duì)于解耦來(lái)說(shuō)是有用的宰翅。

何時(shí)不要去使用工廠模式

當(dāng)被應(yīng)用到錯(cuò)誤的問(wèn)題類(lèi)型上時(shí),這一模式會(huì)給應(yīng)用程序引入大量不必要的復(fù)雜性.除非為創(chuàng)建對(duì)象提供一個(gè)接口是我們編寫(xiě)的庫(kù)或者框架的一個(gè)設(shè)計(jì)上目標(biāo),否則我會(huì)建議使用明確的構(gòu)造器,以避免不必要的開(kāi)銷(xiāo)弃甥。

由于對(duì)象的創(chuàng)建過(guò)程被高效的抽象在一個(gè)接口后面的事實(shí),這也會(huì)給依賴于這個(gè)過(guò)程可能會(huì)有多復(fù)雜的單元測(cè)試帶來(lái)問(wèn)題。

抽象工廠

了解抽象工廠模式也是非常實(shí)用的,它的目標(biāo)是以一個(gè)通用的目標(biāo)將一組獨(dú)立的工廠進(jìn)行封裝.它將一堆對(duì)象的實(shí)現(xiàn)細(xì)節(jié)從它們的一般用例中分離汁讼。

抽象工廠應(yīng)該被用在一種必須從其創(chuàng)建或生成對(duì)象的方式處獨(dú)立,或者需要同多種類(lèi)型的對(duì)象一起工作,這樣的系統(tǒng)中淆攻。

簡(jiǎn)單且容易理解的例子就是一個(gè)發(fā)動(dòng)機(jī)工廠,它定義了獲取或者注冊(cè)發(fā)動(dòng)機(jī)類(lèi)型的方式.抽象工廠會(huì)被命名為AbstractVehicleFactory.抽象工廠將允許像"car"或者"truck"的發(fā)動(dòng)機(jī)類(lèi)型的定義,并且構(gòu)造工廠將僅實(shí)現(xiàn)滿足發(fā)動(dòng)機(jī)合同的類(lèi).(例如:Vehicle.prototype.driven和Vehicle.prototype.breakDown)。

var AbstractVehicleFactory = (function () {

    // Storage for our vehicle types
    var types = {};

    return {
        getVehicle: function ( type, customizations ) {
            var Vehicle = types[type];

            return (Vehicle ? new Vehicle(customizations) : null);
        },

        registerVehicle: function ( type, Vehicle ) {
            var proto = Vehicle.prototype;

            // only register classes that fulfill the vehicle contract
            if ( proto.drive && proto.breakDown ) {
                types[type] = Vehicle;
            }

            return AbstractVehicleFactory;
        }
    };
})();

// Usage:

AbstractVehicleFactory.registerVehicle( "car", Car );
AbstractVehicleFactory.registerVehicle( "truck", Truck );

// Instantiate a new car based on the abstract vehicle type
var car = AbstractVehicleFactory.getVehicle( "car" , {
            color: "lime green",
            state: "like new" } );

// Instantiate a new truck in a similar manner
var truck = AbstractVehicleFactory.getVehicle( "truck" , {
            wheelSize: "medium",
            color: "neon yellow" } );
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末嘿架,一起剝皮案震驚了整個(gè)濱河市瓶珊,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌耸彪,老刑警劉巖伞芹,帶你破解...
    沈念sama閱讀 219,490評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異蝉娜,居然都是意外死亡唱较,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)召川,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)南缓,“玉大人,你說(shuō)我怎么就攤上這事荧呐∥鞴裕” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,830評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵坛增,是天一觀的道長(zhǎng)获雕。 經(jīng)常有香客問(wèn)我,道長(zhǎng)收捣,這世上最難降的妖魔是什么届案? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,957評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮罢艾,結(jié)果婚禮上楣颠,老公的妹妹穿的比我還像新娘。我一直安慰自己咐蚯,他們只是感情好童漩,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,974評(píng)論 6 393
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著春锋,像睡著了一般矫膨。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,754評(píng)論 1 307
  • 那天侧馅,我揣著相機(jī)與錄音危尿,去河邊找鬼。 笑死馁痴,一個(gè)胖子當(dāng)著我的面吹牛谊娇,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播罗晕,決...
    沈念sama閱讀 40,464評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼济欢,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了小渊?” 一聲冷哼從身側(cè)響起法褥,我...
    開(kāi)封第一講書(shū)人閱讀 39,357評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎粤铭,沒(méi)想到半個(gè)月后挖胃,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,847評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡梆惯,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,995評(píng)論 3 338
  • 正文 我和宋清朗相戀三年酱鸭,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片垛吗。...
    茶點(diǎn)故事閱讀 40,137評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡凹髓,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出怯屉,到底是詐尸還是另有隱情蔚舀,我是刑警寧澤,帶...
    沈念sama閱讀 35,819評(píng)論 5 346
  • 正文 年R本政府宣布锨络,位于F島的核電站赌躺,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏羡儿。R本人自食惡果不足惜礼患,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,482評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望掠归。 院中可真熱鬧缅叠,春花似錦、人聲如沸虏冻。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,023評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)厨相。三九已至领曼,卻和暖如春鸥鹉,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背悯森。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,149評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工宋舷, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留绪撵,地道東北人瓢姻。 一個(gè)月前我還...
    沈念sama閱讀 48,409評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像音诈,于是被迫代替她去往敵國(guó)和親幻碱。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,086評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 工廠模式是我們最常用的實(shí)例化對(duì)象模式了细溅,是用工廠方法代替new操作的一種模式褥傍。通常我們所說(shuō)的工廠模式是指工廠方法模...
    zfylin閱讀 1,316評(píng)論 0 7
  • 工廠模式概述 工廠模式是設(shè)計(jì)模式的一種,從功能上來(lái)說(shuō)喇聊,它的主要作用是創(chuàng)建對(duì)象恍风。細(xì)分一下,有三種不同類(lèi)型的工廠模式誓篱,...
    flionel閱讀 705評(píng)論 5 3
  • 這是16年5月份編輯的一份比較雜亂適合自己觀看的學(xué)習(xí)記錄文檔,今天18年5月份再次想寫(xiě)文章窜骄,發(fā)現(xiàn)簡(jiǎn)書(shū)還為我保存起的...
    Jenaral閱讀 2,762評(píng)論 2 9
  • 2017.5.16 每天上下班穿梭在6號(hào)線和5號(hào)線之間锦募,經(jīng)歷人山人海邻遏,樓上樓下糠亩,晴天雨天准验。 正常工作日的時(shí)候,每天...
    xiaokuzi閱讀 189評(píng)論 0 0
  • 美麗的雪谷济似,魅力無(wú)限,潔白的雪花砰蠢,飄飄落落蓖扑。喜歡大自然嗎,喜歡就來(lái)吧台舱!約你律杠,走起潭流!我在洪波客棧等你呦。
    雪谷秋姐閱讀 427評(píng)論 0 4