01 设计模式自学笔记(Java)-静态工厂/简单工厂[创建型模式]

一、简单工厂模式的本质

简单工厂模式的本质是由一个工厂类根据传入的参数,动态决定应该创建哪一个产品类(这些产品类继承自一个父类或接口)的实例。 简单工厂模式的核心,它负责实现创建所有实例的内部逻辑。 工厂类的创建产品类的方法可以被外界直接调用,创建所需的产品对象。 简单工厂模式所创建的所有对象的父类,它负责描述所有实例所共有的公共接口。

二、简单工厂模式目的

将“类实例化的操作”与“使用对象的操作”分开,让使用者不用知道具体参数就可以实例化出所需要的“产品”类,从而避免了在客户端代码中显式指定,实现了解耦。 即使用者可直接消费产品而不需要知道其生产的细节。

三、示例场景

某个电脑公司的在线购物网站可售卖不同型号的笔记本电脑,网站需要为客户提供电脑的配置信息查看功能,每当用户想查看某个型号的电脑配置时,需要给出该型号电脑的CPU、内存大小、硬盘类型和大小、生产日期以及价格等信息。

注意事项:该公司可能每季度都有新型号的电脑上架。

四、代码实现及对比

4.1 原始实现方法

(1)主要实现思路

按照需要创建不同产品的类(X1和X1Pro两个类),通过在客户端程序中使用new关键字创建具体的产品类的实例。

(2)示例代码

X1Pro类定义

public class X1Pro implements IThinkpad{
    String cpu;
    int memory;
    int hardDriveCapacity;
    String hardDriveType;

    /*属性的getter、setter以及toString方法省略*/

    public String viewConfigInfo() {
        return "Thinkpad X1 Pro 配置信息如下:\n--"+cpu + "\n--" + memory + "\n--" + hardDriveCapacity + "\n--" + hardDriveType;
    }
}

X1类定义

public class X1 implements IThinkpad{
    String cpu;
    int memory;
    int hardDriveCapacity;
    String hardDriveType;

    /*属性的getter、setter以及toString方法省略*/

    public String viewConfigInfo() {
        return "Thinkpad X1 配置信息如下:\n--"+cpu + "\n--" + memory + "\n--" + hardDriveCapacity + "\n--" + hardDriveType;
    }
}

客户端类定义

public class OriginalMain {
    public static void main(String[] args) {
        /*原始方法创建使用X1和X1Pro类*/
        X1 x1 = new X1();
        x1.setCpu("英特尔 i7 10570U ");
        x1.setMemory(32);
        x1.setHardDriveType("Seagate SSD");
        x1.setHardDriveCapacity(1);
        System.out.println(x1.viewConfigInfo());

        X1Pro x1Pro = new X1Pro();
        x1Pro.setCpu("英特尔 i9 13770H ");
        x1Pro.setMemory(64);
        x1Pro.setHardDriveType("Seagate SSD");
        x1Pro.setHardDriveCapacity(2048);
        System.out.println(x1Pro.viewConfigInfo());
    }
}

4.2 工厂模式实现方法

X1Pro类定义同上。

X1类定义(X1类换一种方式,使用X1自带的创建方法createThinkPad()创建X1产品类实例)

public class X1 implements IThinkpad{
    String cpu;
    int memory;
    int hardDriveCapacity;
    String hardDriveType;

    /*属性的getter、setter以及toString方法省略*/

    public String viewConfigInfo() {
        return "Thinkpad X1 配置信息如下:\n--"+cpu + "\n--" + memory + "\n--" + hardDriveCapacity + "\n--" + hardDriveType;
    }
}

产品工厂类定义

public class ThinkpadFactory {
    public static IThinkpad creatThinkpad(String model){
        switch (model) {
            case "X1":
                X1 x1 = new X1();
                x1.createThinkPad("X1");
                return x1;
            case "X1Pro":
                X1Pro x1Pro = new X1Pro();
                x1Pro.setCpu("英特尔 i9 13770H ");
                x1Pro.setMemory(64);
                x1Pro.setHardDriveType("Seagate SSD");
                x1Pro.setHardDriveCapacity(2048);
                return x1Pro;
            default:
                throw new RuntimeException("Having no thinkpad of this model.");
        }
    }
}

客户端类定义

public class SimpleFactoryMain {
    public static void main(String[] args) {
        /*使用简单工厂模式创建使用X1和X1Pro类*/
        IThinkpad x11 = ThinkpadFactory.creatThinkpad("X1");
        System.out.println(x11.viewConfigInfo());
        IThinkpad x1p = ThinkpadFactory.creatThinkpad("X1Pro");
        System.out.println(x1p.viewConfigInfo());
    }
}

4.3 二者对比

可以看出,原始方法需要客户端创建产品类的实例对象,简单工厂模式下的客户端只需要创建产品工厂,然后由产品工厂来创建具体产品类的实例对象。这样避免了客户端直接使用new关键字创建具体的产品类实例(如X1或者X1Pro),这样降低了客户端对产品类的细节的了解,符合迪米特法则(即一个对象应当对其它对象有尽可能少的了解)。

在简单工厂模式下工厂类实现的时候,可以选择不同的方法来构建具体的产品类实例:1)产品工厂类内部调用产品类的创建方法实现产品制作(产品属性的赋值),如X1类的创建;2)产品工厂自己实现产品制作,如X1Pro类的创建。

五、简单工厂的UML类图

简单工厂通用UML类图

简单工厂中的角色:

(1)抽象产品类:抽象定义产品的通用属性和方法,可以用Java中的interface或者abstract class来实现;

(2)具体产品类:不同产品具有不同的属性值和方法实现;

(3)简单工厂类:负责按照客户端的“要求”(如参数、类的类型等),提供创建具体产品类的方法,产品类的创建(即new)在此类中实现并返回抽象产品类型的具体产品实例;

(4)客户端类:创建简单工厂类并向其传入响应产品的参数,接受产品工厂返回的产品类实例对象。

六、简单工厂的优缺点

(1)优点

  • 封装,帮助封装简单工厂虽然很简单,但是非常友好地帮助我们实现了组件的封装,然后让组件外部能真正面向接口编程。
  • 解耦,通过简单工厂,实现了客户端和具体实现类的解耦。

如同上面的例子,客户端根本就不知道具体是由那个类实现了具体产品(即,感知不到X1和X1Pro类的存在),也不知道具体产品是如何制作的(即,感知不到X1和X1Pro类中属性的赋值),客户端只是通过工厂获取它需要的产品对象实例(因为工厂返回的类对象是接口类型的,所以客户端还是无法感知具体的产品类的存在和实现,同时,未来新的产品类的增加也对客户端无影响)。

(2)缺点

  • 可能增加客户端的复杂度,如果通过客户端的参数来选择具体的实现类,那么就必须让客户端能理解各个参数所代表的具体功能和含义,这样会增加客户端使用的难度,也部分暴露了内部实现,这种情况可以选用可配置的方式来实现。
  • 不方便扩展子工厂,简单工厂模式通常使用静态工厂方法,这使得无法由子类继承,造成工厂角色无法形成基于继承的等级结构。不过,通常情况下是不需要为简单工厂创建子类的。(也可不通过静态方法实现)。

(3)问题

当新增具体产品类的时候,必须要修改工厂类中的代码,以实现具体产品类的创建,这就违背了开闭原则。出现这个问题的原因就是这个工厂类的职责过多,即生产X1类产品又生产X1Pro类产品,以及新增的产品类都需要在工厂类中创建,所以只要新增新的产品类必须修改工厂类。

解决这个问题的一个可行思路就是细化工厂类的职责,每个产品类有自己的工厂类,这就是下一篇中说的工厂方法设计模式的核心思想。

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

推荐阅读更多精彩内容