JAVA设计模式(01):创建型-工厂模式【工厂方法模式】(Factory Method)

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 文章为转载:代码为自己写的和改编的。本节主要代码: https://yunpan.cn/cqa24JbQmyxp7 访问密码 b917转载自小若博客:http://blog.csdn.net/lovesomnus/article/details/46634987

简单工厂模式虽然简单,但存在一个很严重的问题。当系统中需要引入新产品时,由于静态工厂方法通过所传入参数的不同来创建不同的产品,这必定要修改工厂类的源代码,将违背“开闭原则”,如何实现增加新产品而不影响已有代码?工厂方法模式应运而生,本文将介绍第二种工厂模式——工厂方法模式。

1 日志记录器的设计

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

     Sunny公司的开发人员通过对该需求进行分析,发现该日志记录器有两个设计要点:

      (1) 需要封装日志记录器的初始化过程,这些初始化工作较为复杂,例如需要初始化其他相关的类,还有可能需要读取配置文件(例如连接数据库或创建文件),导致代码较长,如果将它们都写在构造函数中,会导致构造函数庞大,不利于代码的修改和维护;

      (2) 用户可能需要更换日志记录方式,在客户端代码中需要提供一种灵活的方式来选择日志记录器,尽量在不修改源代码的基础上更换或者增加日志记录方式。

      Sunny公司开发人员最初使用简单工厂模式对日志记录器进行了设计,初始结构如图1所示:

11.jpg

1 基于简单工厂模式设计的日志记录器结构图

      在图1中,LoggerFactory充当创建日志记录器的工厂,提供了工厂方法createLogger()用于创建日志记录器,Logger是抽象日志记录器接口,其子类为具体日志记录器。其中,工厂类LoggerFactory代码片段如下所示:

[java]  view plaincopy

  1. //日志记录器工厂  
  2. public class LoggerFactory {  
  3.    //静态工厂方法  
  4.    public static Logger createLogger(String args) {  
  5.        if(args.equalsIgnoreCase("db")) {  
  6.            //连接数据库,代码省略  
  7.            //创建数据库日志记录器对象  
  8.            Logger logger = new DatabaseLogger();  
  9.            //初始化数据库日志记录器,代码省略  
  10.            return logger;  
  11.        }  
  12.        else if(args.equalsIgnoreCase("file")) {  
  13.            //创建日志文件  
  14.            //创建文件日志记录器对象  
  15.            Logger logger = new FileLogger();  
  16.            //初始化文件日志记录器,代码省略  
  17.            return logger;            
  18.        }  
  19.        else {  
  20.            return null;  
  21.        }  
  22.    }  
  23. }  

      为了突出设计重点,我们对上述代码进行了简化,省略了具体日志记录器类的初始化代码。在LoggerFactory类中提供了静态工厂方法createLogger(),用于根据所传入的参数创建各种不同类型的日志记录器。通过使用简单工厂模式,我们将日志记录器对象的创建和使用分离,客户端只需使用由工厂类创建的日志记录器对象即可,无须关心对象的创建过程,但是我们发现,虽然简单工厂模式实现了对象的创建和使用分离,但是仍然存在如下两个问题:

      (1) 工厂类过于庞大,包含了大量的if…else…代码,导致维护和测试难度增大;

      (2) 系统扩展不灵活,如果增加新类型的日志记录器,必须修改静态工厂方法的业务逻辑,违反了“开闭原则”。

      如何解决这两个问题,提供一种简单工厂模式的改进方案?这就是本文所介绍的工厂方法模式的动机之一。


2 工厂方法模式概述

      在简单工厂模式中只提供一个工厂类,该工厂类处于对产品类进行实例化的中心位置,它需要知道每一个产品对象的创建细节,并决定何时实例化哪一个产品类。简单工厂模式最大的缺点是当有新产品要加入到系统中时,必须修改工厂类,需要在其中加入必要的业务逻辑,这违背了“开闭原则”。此外,在简单工厂模式中,所有的产品都由同一个工厂创建,工厂类职责较重,业务逻辑较为复杂,具体产品与工厂类之间的耦合度高,严重影响了系统的灵活性和扩展性,而工厂方法模式则可以很好地解决这一问题。

      在工厂方法模式中,我们不再提供一个统一的工厂类来创建所有的产品对象,而是针对不同的产品提供不同的工厂,系统提供一个与产品等级结构对应的工厂等级结构。工厂方法模式定义如下:

       工厂方法模式(Factory Method Pattern):定义一个用于创建对象的接口,让子类决定将哪一个类实例化。工厂方法模式让一个类的实例化延迟到其子类。工厂方法模式又简称为工厂模式(Factory Pattern),又可称作虚拟构造器模式(Virtual Constructor Pattern)或多态工厂模式(Polymorphic Factory Pattern)。工厂方法模式是一种类创建型模式。

      工厂方法模式提供一个抽象工厂接口来声明抽象工厂方法,而由其子类来具体实现工厂方法,创建具体的产品对象。工厂方法模式结构如图2所示:

12.jpg

2 工厂方法模式结构图

      在工厂方法模式结构图中包含如下几个角色:

    Product(抽象产品):它是定义产品的接口,是工厂方法模式所创建对象的超类型,也就是产品对象的公共父类。

     ● ConcreteProduct(具体产品):它实现了抽象产品接口,某种类型的具体产品由专门的具体工厂创建,具体工厂和具体产品之间一一对应。

      ● Factory(抽象工厂):在抽象工厂类中,声明了工厂方法(Factory Method),用于返回一个产品。抽象工厂是工厂方法模式的核心,所有创建对象的工厂类都必须实现该接口。

     ● ConcreteFactory(具体工厂):它是抽象工厂类的子类,实现了抽象工厂中定义的工厂方法,并可由客户端调用,返回一个具体产品类的实例。

      与简单工厂模式相比,工厂方法模式最重要的区别是引入了抽象工厂角色,抽象工厂可以是接口,也可以是抽象类或者具体类,其典型代码如下所示:

[java]  view plaincopy

  1. public interface Factory {  
  2.    public Product factoryMethod();  
  3. }  

      在抽象工厂中声明了工厂方法但并未实现工厂方法,具体产品对象的创建由其子类负责,客户端针对抽象工厂编程,可在运行时再指定具体工厂类,具体工厂类实现了工厂方法,不同的具体工厂可以创建不同的具体产品,其典型代码如下所示:

[java]  view plaincopy

  1. public class ConcreteFactory implements Factory {  
  2.    public Product factoryMethod() {  
  3.        return new ConcreteProduct();  
  4.    }  
  5. }  

      在实际使用时,具体工厂类在实现工厂方法时除了创建具体产品对象之外,还可以负责产品对象的初始化工作以及一些资源和环境配置工作,例如连接数据库、创建文件等。

      在客户端代码中,只需关心工厂类即可,不同的具体工厂可以创建不同的产品,典型的客户端类代码片段如下所示:

[java]  view plaincopy

  1. ……  
  2. Factory factory;  
  3. factory = new ConcreteFactory(); //可通过配置文件实现  
  4. Product product;  
  5. product = factory.factoryMethod();  
  6. ……  

      可以通过配置文件来存储具体工厂类ConcreteFactory的类名,更换新的具体工厂时无须修改源代码,系统扩展更为方便。

网络异常,图片无法展示
|

思考

工厂方法模式中的工厂方法能否为静态方法?为什么?


3 完整解决方案

       Sunny公司开发人员决定使用工厂方法模式来设计日志记录器,其基本结构如图3所示:

13.jpg

3 日志记录器结构图

      在图3中,Logger接口充当抽象产品,其子类FileLogger和DatabaseLogger充当具体产品,LoggerFactory接口充当抽象工厂,其子类FileLoggerFactory和DatabaseLoggerFactory充当具体工厂。完整代码如下所示:

[java]  view plaincopy

  1. //日志记录器接口:抽象产品  
  2. public interface Logger {  
  3.    public void writeLog();  
  4. }  
  5.  
  6. //数据库日志记录器:具体产品  
  7. class DatabaseLogger implements Logger {  
  8.    public void writeLog() {  
  9.        System.out.println("数据库日志记录。");  
  10.    }  
  11. }  
  12.  
  13. //文件日志记录器:具体产品  
  14. class FileLogger implements Logger {  
  15.    public void writeLog() {  
  16.        System.out.println("文件日志记录。");  
  17.    }  
  18. }  
  19.  
  20. //日志记录器工厂接口:抽象工厂  
  21. public interface LoggerFactory {  
  22.    public Logger createLogger();  
  23. }  
  24.  
  25. //数据库日志记录器工厂类:具体工厂  
  26. class DatabaseLoggerFactory implements LoggerFactory {  
  27.    public Logger createLogger() {  
  28.            //连接数据库,代码省略  
  29.            //创建数据库日志记录器对象  
  30.            Logger logger = new DatabaseLogger();  
  31.            //初始化数据库日志记录器,代码省略  
  32.            return logger;  
  33.    }    
  34. }  
  35.  
  36. //文件日志记录器工厂类:具体工厂  
  37. class FileLoggerFactory implements LoggerFactory {  
  38.    public Logger createLogger() {  
  39.            //创建文件日志记录器对象  
  40.            Logger logger = new FileLogger();  
  41.            //创建文件,代码省略  
  42.            return logger;  
  43.    }    
  44. }  

      编写如下客户端测试代码:

[java]  view plaincopy

  1. public class Client {  
  2.    public static void main(String args[]) {
  3.        LoggerFactory factory = new FileLoggerFactory(); //可引入配置文件实现  
  4.        Logger logger = factory.createLogger();  
  5.        logger.writeLog();  
  6.    }  
  7. }  

      编译并运行程序,输出结果如下:

文件日志记录。

 

4 反射与配置文件

      为了让系统具有更好的灵活性和可扩展性,Sunny公司开发人员决定对日志记录器客户端代码进行重构,使得可以在不修改任何客户端代码的基础上更换或增加新的日志记录方式。

      在客户端代码中将不再使用new关键字来创建工厂对象,而是将具体工厂类的类名存储在配置文件(如XML文件)中,通过读取配置文件获取类名字符串,再使用Java的反射机制,根据类名字符串生成对象。在整个实现过程中需要用到两个技术:Java反射机制与配置文件读取。软件系统的配置文件通常为XML文件,我们可以使用DOM (Document Object Model)SAX (Simple API for XML)StAX (Streaming API for XML)等技术来处理XML文件。关于DOMSAXStAX等技术的详细学习大家可以参考其他相关资料,在此不予扩展。

网络异常,图片无法展示
|

扩展

关于JavaXML的相关资料,大家可以阅读Tom MyersAlexander Nakhimovsky所著的《Java XML编程指南》一书或访问developer Works中国中的“Java XML 技术专题”,参考链接:

http://www.ibm.com/developerworks/cn/xml/theme/x-java.html

      Java反射(Java Reflection)是指在程序运行时获取已知名称的类或已有对象的相关信息的一种机制,包括类的方法、属性、父类等信息,还包括实例的创建和实例类型的判断等。在反射中使用最多的类是ClassClass类的实例表示正在运行的Java应用程序中的类和接口,其forName(String className)方法可以返回与带有给定字符串名的类或接口相关联的Class对象,再通过Class对象的newInstance()方法创建此对象所表示的类的一个新实例,即通过一个类名字符串得到类的实例。如创建一个字符串类型的对象,其代码如下:

[java]  view plaincopy

  1. //通过类名生成实例对象并将其返回  
  2. Class c=Class.forName("String");  
  3. Object obj=c.newInstance();  
  4. return obj;  

      此外,在JDK中还提供了java.lang.reflect包,封装了其他与反射相关的类,此处只用到上述简单的反射代码,在此不予扩展。

      Sunny公司开发人员创建了如下XML格式的配置文件config.xml用于存储具体日志记录器工厂类类名:

[html]  view plaincopy

  1. <!— config.xml -->  
  2. <?xml version="1.0"?>  
  3. <config>  
  4.    <className>FileLoggerFactory</className>  
  5. </config>  

      为了读取该配置文件并通过存储在其中的类名字符串反射生成对象,Sunny公司开发人员开发了一个名为XMLUtil的工具类,其详细代码如下所示:

[java]view plaincopy

  1. import java.io.File;  
  2. import org.dom4j.Document;  
  3. import org.dom4j.io.SAXReader;  
  4. publicclass XMLUtil {  
  5.    //该方法用于从XML配置文件中提取图表类型,并返回类型名    
  6.    publicstatic Object getBean() throws Exception {  
  7.        SAXReader reader = new SAXReader();  
  8.        String path = XMLUtil.class.getClassLoader().  
  9.                getResource("com/somnus/designPatterns/factoryMethod/config.xml").getPath();  
  10.        Document document = reader.read(new File(path));  
  11.        String cName = document.selectSingleNode("/config/className").getText();  
  12.        //通过类名生成实例对象并将其返回    
  13.        Class<?> c = Class.forName(cName);    
  14.        Object obj = c.newInstance();    
  15.        return obj;    
  16.    }    
  17. }  
1. import java.io.File;
2. import org.dom4j.Document;
3. import org.dom4j.io.SAXReader;
4. public class XMLUtil {
5. //该方法用于从XML配置文件中提取图表类型,并返回类型名  
6. public static Object getBean() throws Exception {
7. SAXReader reader = new SAXReader();
8. String path = XMLUtil.class.getClassLoader().
9.                 getResource("com/somnus/designPatterns/factoryMethod/config.xml").getPath();
10. Document document = reader.read(new File(path));
11. String cName = document.selectSingleNode("/config/className").getText();
12. //通过类名生成实例对象并将其返回  
13.         Class<?> c = Class.forName(cName);  
14. Object obj = c.newInstance();  
15. return obj;  
16.     }  
17. }


      有了XMLUtil类后,可以对日志记录器的客户端代码进行修改,不再直接使用new关键字来创建具体的工厂类,而是将具体工厂类的类名存储在XML文件中,再通过XMLUtil类的静态工厂方法getBean()方法进行对象的实例化,代码修改如下:

[java]view plaincopy

  1. publicclass Client {  
  2.    publicstaticvoid main(String[] args) throws Exception {  
  3.        //getBean()的返回类型为Object,需要进行强制类型转换    
  4.        LoggerFactory factory = (LoggerFactory)XMLUtil.getBean();  
  5.        Logger logger = factory.createLogger();    
  6.        logger.writeLog();    
  7.    }  
  8. }  
1. public class Client {
2. public static void main(String[] args) throws Exception {
3. //getBean()的返回类型为Object,需要进行强制类型转换  
4. LoggerFactory factory = (LoggerFactory)XMLUtil.getBean(); 
5. Logger logger = factory.createLogger();  
6.         logger.writeLog();  
7.     }
8. }


      引入XMLUtil类和XML配置文件后,如果要增加新的日志记录方式,只需要执行如下几个步骤:

      (1) 新的日志记录器需要继承抽象日志记录器Logger

      (2) 对应增加一个新的具体日志记录器工厂,继承抽象日志记录器工厂LoggerFactory,并实现其中的工厂方法createLogger(),设置好初始化参数和环境变量,返回具体日志记录器对象;

      (3) 修改配置文件config.xml,将新增的具体日志记录器工厂类的类名字符串替换原有工厂类类名字符串;

      (4) 编译新增的具体日志记录器类和具体日志记录器工厂类,运行客户端测试类即可使用新的日志记录方式,而原有类库代码无须做任何修改,完全符合“开闭原则”。

     通过上述重构可以使得系统更加灵活,由于很多设计模式都关注系统的可扩展性和灵活性,因此都定义了抽象层,在抽象层中声明业务方法,而将业务方法的实现放在实现层中。

网络异常,图片无法展示
|

思考

       有人说:可以在客户端代码中直接通过反射机制来生成产品对象,在定义产品对象时使用抽象类型,同样可以确保系统的灵活性和可扩展性,增加新的具体产品类无须修改源代码,只需要将其作为抽象产品类的子类再修改配置文件即可,根本不需要抽象工厂类和具体工厂类。

       试思考这种做法的可行性?如果可行,这种做法是否存在问题?为什么?

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
4月前
|
存储 Java 开发者
【Java新纪元启航】JDK 22:解锁未命名变量与模式,让代码更简洁,思维更自由!
【9月更文挑战第7天】JDK 22带来的未命名变量与模式匹配的结合,是Java编程语言发展历程中的一个重要里程碑。它不仅简化了代码,提高了开发效率,更重要的是,它激发了我们对Java编程的新思考,让我们有机会以更加自由、更加创造性的方式解决问题。随着Java生态系统的不断演进,我们有理由相信,未来的Java将更加灵活、更加强大,为开发者们提供更加广阔的舞台。让我们携手并进,共同迎接Java新纪元的到来!
83 11
|
4月前
|
设计模式 Java
Java设计模式-工厂方法模式(4)
Java设计模式-工厂方法模式(4)
|
5月前
|
消息中间件 Java
【实战揭秘】如何运用Java发布-订阅模式,打造高效响应式天气预报App?
【8月更文挑战第30天】发布-订阅模式是一种消息通信模型,发送者将消息发布到公共队列,接收者自行订阅并处理。此模式降低了对象间的耦合度,使系统更灵活、可扩展。例如,在天气预报应用中,`WeatherEventPublisher` 类作为发布者收集天气数据并通知订阅者(如 `TemperatureDisplay` 和 `HumidityDisplay`),实现组件间的解耦和动态更新。这种方式适用于事件驱动的应用,提高了系统的扩展性和可维护性。
82 2
|
5月前
|
设计模式 Java
Java 设计模式之谜:工厂模式与抽象工厂模式究竟隐藏着怎样的神奇力量?
【8月更文挑战第30天】在Java编程中,设计模式为常见问题提供了高效解决方案。工厂模式与抽象工厂模式是常用的对象创建型设计模式,能显著提升代码的灵活性、可维护性和可扩展性。工厂模式通过定义创建对象的接口让子类决定实例化哪个类;而抽象工厂模式则进一步提供了一个创建一系列相关或相互依赖对象的接口,无需指定具体类。这种方式使得系统更易于扩展和维护。
46 1
|
5月前
|
Java
"揭秘Java IO三大模式:BIO、NIO、AIO背后的秘密!为何AIO成为高并发时代的宠儿,你的选择对了吗?"
【8月更文挑战第19天】在Java的IO编程中,BIO、NIO与AIO代表了三种不同的IO处理机制。BIO采用同步阻塞模型,每个连接需单独线程处理,适用于连接少且稳定的场景。NIO引入了非阻塞性质,利用Channel、Buffer与Selector实现多路复用,提升了效率与吞吐量。AIO则是真正的异步IO,在JDK 7中引入,通过回调或Future机制在IO操作完成后通知应用,适合高并发场景。选择合适的模型对构建高效网络应用至关重要。
101 2
|
5月前
|
设计模式 XML 存储
【二】设计模式~~~创建型模式~~~工厂方法模式(Java)
文章详细介绍了工厂方法模式(Factory Method Pattern),这是一种创建型设计模式,用于将对象的创建过程委托给多个工厂子类中的某一个,以实现对象创建的封装和扩展性。文章通过日志记录器的实例,展示了工厂方法模式的结构、角色、时序图、代码实现、优点、缺点以及适用环境,并探讨了如何通过配置文件和Java反射机制实现工厂的动态创建。
【二】设计模式~~~创建型模式~~~工厂方法模式(Java)
|
5月前
|
设计模式 XML Java
【一】设计模式~~~创建型模式~~~简单工厂模式(Java)
文章详细介绍了简单工厂模式(Simple Factory Pattern),这是一种创建型设计模式,用于根据输入参数的不同返回不同类的实例,而客户端不需要知道具体类名。文章通过图表类的实例,展示了简单工厂模式的结构、时序图、代码实现、优缺点以及适用环境,并提供了Java代码示例和扩展应用,如通过配置文件读取参数来实现对象的创建。
【一】设计模式~~~创建型模式~~~简单工厂模式(Java)
|
5月前
java.lang.IllegalStateException: Could not find method onClickcrea(View) in a parent or ancestor Con
java.lang.IllegalStateException: Could not find method onClickcrea(View) in a parent or ancestor Con
68 1
|
4月前
|
JSON Java UED
uniapp:使用DCloud的uni-push推送消息通知(在线模式)java实现
以上展示了使用Java结合DCloud的uni-push进行在线消息推送的基本步骤和实现方法。实际部署时,可能需要依据实际项目的规模,业务场景及用户基数进行必要的调整和优化,确保消息推送机制在保证用户体验的同时也满足业务需求。
252 0
|
6月前
|
存储 Java 编译器
Java面试题:描述方法区(Method Area)的作用以及它在JVM中的演变(从永久代到元空间)
Java面试题:描述方法区(Method Area)的作用以及它在JVM中的演变(从永久代到元空间)
75 3