Java设计模式-创建型模式-工厂方法模式

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

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

1 概述

工厂方法模式简称为工厂模式(Factory Pattern),又可称为虚拟构造器模式(Virtual Constructor Pattern)或多态工厂模式(Polymorphic Factory Pattern)。工厂方法模式是一种创建型模式,在工厂方法模式中,工厂父类负责定义创建产品对象的公共接口,而工厂子类负责生成具体的产品对象,这样做的目的时将产品类的实例化操作延迟到工厂子类中完成,即通过工厂子类来确定究竟应该实例化哪一个具体的产品类。

工厂方法模式:定义一个用于创建对象的接口,但是让子类决定将哪一个类实例化。工厂方法模式让一个类的实例化延迟到其子类。

2 结构与实现

2.1 工厂方法模式结构

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

  1. AbstractProduct(抽象产品):它是定义产品的接口,是工厂方法模式所创建对象的超类,是具体产品对象的公共父类。可以是抽象类或者接口。
  2. ConcreteProduct(具体产品):它实现了抽象产品所定义的接口,某种类型的具体产品由专门的具体工厂创建,<span style="color:red;font-weight:900">具体工厂和具体产品之间一一对应</span>。
  3. AbstractFactory(抽象工厂):在抽象工厂中声明了抽象工厂方法,用于返回一个产品。抽象工厂是工厂方法模式的核心,所有创建对象的工厂类都必须实现该接口。
  4. ConcreteFactory(具体工厂):它是抽象工厂类的子类,实现了在抽象工厂中声明的工厂方法,并可由客户端调用,返回一个具体产品类的实例。

2.2 工厂方法模式举例

一、背景介绍

某系统运行日志记录器(Logger)可以通过多种途径保存系统的运行日志,例如通过文件记录或数据库记录,用户可以通过修改配置文件灵活地更换日志记录方式。在设计各类日志记录器时,开发人员发现需要对日志记录器时,开发人员发现需要对日志记录器进行一些初始化工作,初始化参数地设置过程较为复杂,而且某些参数地设置有严格地先后次序 ,否则可能会发生记录失败。

为了更好地封装记录器的初始化过程并保证多种记录器切换的灵活性,现使用工厂方法模式设计该系统(注:在Java中常用的日志记录工具有SLF4J、Log4j、GCLogViewer、Logstash等。)

二、项目结构

项目结构.png

三、抽象产品

public interface Logger {
    public void writeLog();
}

四、具体产品

具体日志记录器:数据库日志记录器

public class DatabaseLogger implements Logger {
    @Override
    public void writeLog() {
        System.out.println("数据库日志记录");
    }
}

具体日志记录器:文件日志记录器

public class FileLogger implements Logger {
    @Override
    public void writeLog() {
        System.out.println("文件日志记录");
    }
}

五、抽象工厂

public interface LoggerFactory {
    public Logger createLogger();
}

六、具体工厂

具体工厂:数据库日志记录器工厂

public class DatabaseLoggerFactory implements LoggerFactory {
    @Override
    public Logger createLogger() {
        //连接数据库,创建数据库日志记录对象等操作,代码省略
        Logger logger = new DatabaseLogger();
        //写入日志等操作,代码省略
        return logger;
    }
}

具体工厂:文件日志记录器工厂

public class FileLoggerFactory implements LoggerFactory {
    @Override
    public Logger createLogger() {
        //创建文件日志记录器对象,代码省略
        Logger logger = new FileLogger();
        //创建文件,代码省略
        return logger;
    }
}

七、测试Client

import CreationalPattern.FactoryMethodPattern.AbstractFactory.LoggerFactory;
import CreationalPattern.FactoryMethodPattern.AbstractProduct.Logger;
import CreationalPattern.FactoryMethodPattern.ConcreteFactory.DatabaseLoggerFactory;
import CreationalPattern.FactoryMethodPattern.ConcreteFactory.FileLoggerFactory;

public class Client {
    public static void main(String[] args) {
        LoggerFactory factory;
        Logger logger;
        /**
         * note:可以引入文件读取和反射加载机制,提供一个专门的配置文件。
         * note:只需要修改具体的XxxLoggerFactory是哪一个即可实现生产两种不同的Logger
         */
        factory = new FileLoggerFactory();//文件日志记录
//        factory = new DatabaseLoggerFactory();//数据库日志记录
        logger = factory.createLogger();
        logger.writeLog();
    }
}

八、打印结果

我们只需要改变实现抽象工厂的具体工厂类型,即可创建不同的日志记录器:


创建数据库日志记录器.png
创建文件日志记录器.png

3 总结

工厂方法模式是简单工厂模式的延申,它继承了简单工厂模式的优点,同时还弥补了简单工厂模式的不足。工厂方法模式是使用频率最高的设计模式之一,是很多开源框架和API类库的核心模式。

3.1 工厂方法模式优点

  1. 在工厂方法模式中,工厂方法用来创建客户所需要的产品,同时还向客户隐藏了哪种具体产品将被实例化这一细节,用户只需要关心所需产品对应的工厂,无需关心创建细节,甚至无需知道具体产品类的类名。
  2. 基于工厂角色和产品角色的多态性设计是工厂方法模式的关键。它能够 让工厂自主确定创建何种产品对象,而如何创建这个对象的细节完全封装在具体工厂内部。工厂方法模式之所以被称为多态工厂模式,正式因为所有的具体的工厂类都具有同一抽象父类。
  3. 使用工厂方法模式的另一个优点是在系统中加入新产品时无需修改抽象工厂和抽象产品提供的接口,无需修改客户端,也无需修改其他的具体工厂和具体产品,而只需添加一个具体工厂和具体产品即可,这样系统的可扩展性也就变得非常好,完全符合开闭原则。

3.2 工厂方法模式确定

  1. 在添加新产品时需要编写新的具体产品类,而且还要提供与之对应的具体工厂类,系统中类的个数成对增加,在一定程度上增加了系统的复杂度,有更多的类需要编译和运行,会给系统带来额外的开销。
  2. 由于考虑到徐通的可扩展性,需要引入抽象层,在客户端代码中均使用抽象层进行定义,增加了系统的抽象性和理解难度。

3.3 工厂方法模式适用环境

  1. 客户端不知道它所需要的对象的类。在工厂方法模式中,客户端不需要知道具体产品类的类名,只需要知道对应的工厂即可,具体产品对象有具体工厂类创建,可将具体工厂 类的类名存储在配置文件或数据库中。
  2. 抽象工厂类通过其子类来指定创建哪个对象。在工厂方法模式中,对于抽象工厂类只需要提供一个创建产品的接口,而由子类来确定具体要创建的对象,利用面向对象的多态性和里氏替换原则,在程序运行时子类对象将覆盖父类对象,从而使得系统更容易扩展。

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

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

推荐阅读更多精彩内容