Typescript 工厂模式

标签: 前端 设计模式 工厂模式 typescript factory


如果下面的代码你能轻易阅读,那么你已经熟悉工厂模式,可以接着学习其他的设计模式。

factory.jpg

工厂模式:一个类或对象中往往会包含别的对象,在创建这种成员对象时。我们经常使用new来创建,但是这会导致相关的两个类之间产生依赖性。工厂模式使用了一个方法来决定使用究竟实例哪个类。

简单工厂

工厂模式,即类似于工厂一样有一个流程化的形式。
比方说服装厂,先是任意一种服装的制作,然后是漂洗,最后是包装出场。
写成代码如下所示。

/* simple-factory.ts */
import CloseTypeEnum from './close-type-enum';
class SimpleFactory {
    private static _instance: SimpleFactory;
    public static get instance() {
        if (!this._instance) {
            this._instance = new SimpleFactory();
        }
        return this._instance;
    }
    public createClose(closeType: CloseTypeEnum): string {
        let closeName: string = null;
        switch (closeType) {
            case CloseTypeEnum.Sweater:
                closeName = 'sweater';
                break;
            case CloseTypeEnum.Shirt:
                closeName = 'shirt';
                break;
            case CloseTypeEnum.Jacket:
                closeName = 'jacket';
                break;
            default :
                throw new Error('can\'t find type' + closeType);
        }
        this.washingsClose(closeName);
        this.packagingClose(closeName);
        return closeName;
    }
    public washingsClose(close: string) {}
    public packagingClose(close: string) {}
}

用一个类来创建经过一系列流程对象,对象的创建和流程解除依赖。从而解除依赖,更容易增加对象种类。便于以后维护和扩展。

抽象工厂

如上面的服装厂的流程,我们的服装厂要增加一种衣服的类型时,仍然需要进入工厂类中增加一个case。如果每一个case里面的流程是非常复杂的,那我们用switch case显然是不合适的。
这时候我们需要用到抽象工厂
将生产衣服的这个switch case抽象成一个接口,因为我们只关心最终造出来的衣服,而不关心制作的流程。
换成代码就是写几个方法都去实现makeClose的接口。

/* i-factory-interface.ts */
interface ICloseMakerFactory {
    makeClose(): IClose
}
interface IClose {
    name: string
    price: number
    note: string
}
export {
    ICloseMakerFactory,
    IClose,
};
/* sweater-producer.ts */
import { IClose, ICloseMakerFactory } from '../i-factory-interface';
export default class SweaterProducer implements ICloseMakerFactory {
    makeClose(): IClose {
        return {
            name: 'sweater',
            price: 15,
            note: 'The process can be complex',
        };
    }
}
/* shirt-producer.ts */
import { IClose, ICloseMakerFactory } from '../i-factory-interface';
export default class ShirtProducer implements ICloseMakerFactory {
    makeClose(): IClose {
        return {
            name: 'shirt',
            price: 20,
            note: 'The process can be complex',
        };
    }
}
**/* jacket-producer.ts */
import { IClose, ICloseMakerFactory } from '../i-factory-interface';
export default class JacketProducer implements ICloseMakerFactory {
    makeClose(): IClose {
        return {
            name: 'jacket',
            price: 25,
            note: 'The process can be complex',
        };
    }
}**

然后在主方法中调用这些方法去实现生产衣服的过程。

class AbstractFactory {
    private static _instance: AbstractFactory;
    private closeTypeAnalyzerMapping: Map<CloseTypeEnum, ICloseMakerFactory>;
    constructor() {
        this.closeTypeAnalyzerMapping.set(CloseTypeEnum.Jacket, new JacketProducer());
        this.closeTypeAnalyzerMapping.set(CloseTypeEnum.Shirt, new ShirtProducer());
        this.closeTypeAnalyzerMapping.set(CloseTypeEnum.Sweater, new SweaterProducer());
    }
    public static get instance() {
        if (!this._instance) {
            this._instance = new AbstractFactory();
        }
        return this._instance;
    }
    public makeCloseTypeAnalyzer(type: CloseTypeEnum): ICloseMakerFactory {
        if (this.closeTypeAnalyzerMapping.has(type)) {
            return this.closeTypeAnalyzerMapping.get(type);
        } else {
            throw new Error('our factory don\'t have this type');
        }
    }
}
const shrit= AbstractFactory.instance.makeCloseTypeAnalyzer(CloseTypeEnum.Shirt).makeClose();

当然,实际场景中过程的每一步都会根据衣服的种类有多种实现方法,我们可以按照生产衣服一样将他们区分开来。
这样,未来我们增加了一种衣服的类型或者洗涤方法等,只需要增加一个实现类而不用大面积的改变主方法。

工厂模式的利弊

利:

  • 消除对象间的耦合:
    使用工厂方法而不是new关键字,解除创建过程中的耦合步骤,增加可维护性和可测试性。
  • 易于模块化
    使用工厂模式,可以先创建一个抽象的父类,然后在子类中创建工厂方法,从而把成员对象的实例化推迟到更专门的子类中进行。增加可拓展性。

弊:
工厂模式在处理多流程或者是根据上下文创建实例的情况下非常好用。但是在单一流程或者是不需要根据上下文创建对象的情况下使用反而会增加阅读成本。

所以我们应该在恰当的场景中使用工厂模式。如果拿不定主意,那就不要使用,因为在以后的重构中还有机会使用工厂模式。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,546评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,224评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,911评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,737评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,753评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,598评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,338评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,249评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,696评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,888评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,013评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,731评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,348评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,929评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,048评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,203评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,960评论 2 355

推荐阅读更多精彩内容

  • 设计模式概述 在学习面向对象七大设计原则时需要注意以下几点:a) 高内聚、低耦合和单一职能的“冲突”实际上,这两者...
    彦帧阅读 3,747评论 0 14
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,657评论 18 139
  • 发了一会呆,回忆就涌了上来,画画是回忆的最好记录!!
    小怪兽打大boss阅读 274评论 0 0
  • 怎样为AppleWatch设计App 先讲一下AppleWatch设备本身的特点吧,手表是很私人化的物品,对用户来...
    大大的鱼阅读 1,708评论 0 7
  • 想来我们二十一世纪的第一批九零后马上也要步入三十岁高龄了,身边这些百米冲刺一般奔三的大龄单身狗们已经开始有些急不可...
    九作阅读 291评论 0 1