Java设计模式-创建型模式-抽象工厂模式

此系列文章为清华大学出版社出版刘伟编著《Java设计模式》的学习笔记。

>>全部23种设计模式<<

1 概述

抽象工厂模式又成为工具(Kit)模式,它是一种对象创建型模式。

1.1 抽象工厂模式概念

提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。

1.2 抽象工厂模式与工厂方法模式的区别

抽象工厂模式是所有形式的工厂模式中最为抽象和最具一般性的一种形式。抽象工厂模式与工厂方法模式的最大区别在于,工厂方法模式针对的是一个产品等级结构,而抽象工厂模式需要面对多个产品等级结构,一个工厂等级结构可以负责多个不同产品等级结构中的产品对象的创建。当一个工厂等级结构可以创建出分属于不同产品等级结构的一个产品族中的所有对象时,抽象工厂模式比工厂方法模式更为简单、更有效率。

2 结构与实现

2.1 抽象工厂模式结构

抽象工厂模式包含4个角色

  1. AbstractFactory(抽象工厂):它声明了一组用于创建一族产品的方法,每一个方法对应一种产品。
  2. ConcreteFactory(具体工厂):它实现了在抽象工厂中声明的创建产品的方法,生成一组具体产品,这些产品构成一个产品族,每一个产品都位于某个产品等级结构中。
  3. AbstractProduct(抽象产品):它为每种产品声明接口,在抽象产品中声明了产品所具备的业务方法。
  4. ConcreteProduct(具体产品):它定义具体工厂生产的具体产品对象,实现抽象产品接口中声明的业务方法。

2.2 抽象工厂模式举例

一、背景介绍

某软件公司要开发一套界面皮肤库,可以基于Java的桌面软件进行界面美化。用户在使用时可以通过菜单来选择皮肤,不同的皮肤将提供不同的组件视觉效果。组件包括按钮(Button)、文本框(TextField)、组合框(ComboBox)等界面元素,皮肤分为 春天系皮肤和夏天系皮肤。要求该皮肤库具备良好的灵活性和可扩展性,用户可以自由选择不同的皮肤,开发人员可以在不修改既有代码的基础上增加新的皮肤,请使用抽象工厂模式来设计。

二、项目结构

抽象工厂模式结构.png

三、抽象产品

我们需要生产皮肤包括的组件有按钮(Button)、文本框(TextField)、组合框(ComboBox)。即使皮肤系列不同,它们提供的功能类型是一致的,只是视觉上不同,对功能的实现不同。所以有三个抽象接口Button、TextField、ComboBox。

public interface Button {
    public void display();
}
public interface TextField {
    public void display();
}
public interface ComboBox {
    public void display();
}

四、抽象工厂

我们面向接口编程,抽象工厂中创建的对象是抽象的对象。

public interface SkinFactory {
    public Button createButton();
    public TextField createTextField();
    public ComboBox createComboBox();
}

五、具体产品

春天系皮肤组件

public class SpringButton implements Button {
    @Override
    public void display() {
        System.out.println("春天系皮肤【按钮】");
    }
}
public class SpringTextField implements TextField {
    @Override
    public void display() {
        System.out.println("春天系皮肤【文本框】");
    }
}
public class SpringComboBox implements ComboBox {
    @Override
    public void display() {
        System.out.println("春天系皮肤【组合框】");
    }
}

夏天系皮肤组件

public class SummerButton implements Button {
    @Override
    public void display() {
        System.out.println("夏天系皮肤【按钮】");
    }
}
public class SummerTextField implements TextField {
    @Override
    public void display() {
        System.out.println("夏天系皮肤【文本框】");
    }
}
public class SummerComboBox implements ComboBox {
    @Override
    public void display() {
        System.out.println("夏天系皮肤【组合框】");
    }
}

六、具体工厂

春天皮肤工厂

public class SpringSkinFactory implements SkinFactory {
    @Override
    public Button createButton() {
        return new SpringButton();
    }
    @Override
    public TextField createTextField() {
        return new SpringTextField();
    }
    @Override
    public ComboBox createComboBox() {
        return new SpringComboBox();
    }
}

夏天皮肤工厂

public class SummerSkinFactory implements SkinFactory {
    @Override
    public Button createButton() {
        return new SummerButton();
    }
    @Override
    public TextField createTextField() {
        return new SummerTextField();
    }
    @Override
    public ComboBox createComboBox() {
        return new SummerComboBox();
    }
}

七、XML确定皮肤类型

编写config.xml

<?xml version="1.0" encoding="UTF-8"?>
<config>
    <className>CreationalPattern.AbstractFactoryPattern.ConcreteFactory.SpringSkinFactory</className>
    <!--<className>CreationalPattern.AbstractFactoryPattern.ConcreteFactory.SummerSkinFactory</className>-->
</config>

八、编写XMLUtil

import org.w3c.dom.Document;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;

public class XMLUtil {
    public static Object getBean(){
        try {
            //todo:创建 DOM 文档对象
            DocumentBuilderFactory documentBuilderFactory= DocumentBuilderFactory.newInstance();
            DocumentBuilder documentBuilder = documentBuilderFactory.newDocumentBuilder();
            Document doc;
            doc = documentBuilder.parse("src\\CreationalPattern\\AbstractFactoryPattern\\config.xml");
            //todo:获取包含类名的文本节点
            NodeList nodeList = doc.getElementsByTagName("className");
            Node classNode = nodeList.item(0).getFirstChild();
            String cName = classNode.getNodeValue();
            //todo:通过类名生成实例对象并将其返回
            Class clz = Class.forName(cName);
            Object obj = clz.newInstance();
            return obj;
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    }
}

九、测试Client

public class Client {
    public static void main(String[] args) {
        //todo:抽象层定义
        SkinFactory skinFactory;
        Button button;
        TextField textField;
        ComboBox comboBox;
        skinFactory = (SkinFactory) XMLUtil.getBean();
        //todo:开始生产
        button = skinFactory.createButton();
        textField = skinFactory.createTextField();
        comboBox = skinFactory.createComboBox();
        //todo:检验生产结果
        button.display();
        textField.display();
        comboBox.display();
    }
}

十、打印结果

当config.xml中的className节点值为CreationalPattern.AbstractFactoryPattern.ConcreteFactory.SpringSkinFactory时,抽象工厂类的实现类为SpringFactory,生产的组件检验结果如下:

春天系工厂生产组件检验.png

当config.xml中的className节点值为CreationalPattern.AbstractFactoryPattern.ConcreteFactory.SummerSkinFactory时,抽象工厂类的实现类为SummerFactory,生产的组件检验结果如下:

夏天系工厂生产组件检验.png

3 总结

3.1 实现过程梳理

在测试的main()方法中,首先定义了所有的顶级抽象对象,抽象的skinFactory、抽象的button、抽象的textField、抽象的comboBox。然后,我们通过读取静态的XML文件,获取具体工厂的实现,是春天系皮肤工厂或是夏天系皮肤工厂,这里的XML文件仅仅是提供类名的一个作用,完全可以用字符串直接完成。具体的工厂生产的是具体的组件,调用他们的方法,将产生不同的效果。

3.2 抽象工厂模式优点

  1. 抽象工厂模式隔离了具体类的生成,使得客户端并不需要知道什么被创建。由于这种隔离,更换一个具体工厂变得容易,所有的具体工厂都实现了抽象工厂中定义的接口方法,因此只需要改变具体工厂的实例就可以在某种程度上改变整个软件系统的行为细节。
  2. 当一个产品族中的多个对象被设计成一起工作时,它能够保证客户端始终只使用同一个产品族中的对象。
  3. 增加新的产品族很方便,无需修改已有系统,符合开闭原则。

3.3 抽象工厂模式的缺点

  • 抽象工厂模式的缺点是:增加新的产品结构很麻烦,需要对原有系统进行较大的修改,甚至需要修改抽象层代码,违背了开闭原则。

3.4 抽象工厂模式使用场景

  1. 一个系统不应该依赖于产品类实例如何被创建、组合和表达的细节,这对于所有类型的工厂模式都是很重要的,用户无需关心对象的创建过程,将对象的创建和使用解耦。
  2. 系统中有多于一个的产品族,而每次只使用其中某一个产品族。可以通过配置文件等方式来使用户能够动态改变产品族,也可以很方便地增加新的产品族。
  3. 属于同一个产品族地产品将在一起使用,这一约束必须在系统地设计中体现出来。同一个产品族中的产品可以是没有任何关系的对象,但是它们都具有一些共同的约束,如同易操作系统下的按钮和文本框,按钮与文本框之间没有直接关系,但它们都是属于某一操作系统的,此时具有一个共同的约束条件,及操作系统的类型。
  4. 产品等级结构稳定,在设计完成之后不会向系统中增加新的产品等级结构或者删除已经有的产品等级结构。

>>全部23种设计模式<<

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

推荐阅读更多精彩内容