工厂顾名思义就是创建产品,根据产品是具体产品还是具体工厂可分为简单工厂模式和工厂方法模式,根据工厂的抽象程度可分为工厂方法模式和抽象工厂模式。该模式用于封装和管理对象的创建,是一种创建型模式。本文从一个具体的例子逐步深入分析,来体会三种工厂模式的应用场景和利弊。
1. 简单工厂模式
该模式对对象创建管理方式最为简单,因为其仅仅简单的对不同类对象的创建进行了一层薄薄的封装。该模式通过向工厂传递类型来指定要创建的对象,其UML类图如下:![[设计模式] 工厂模式 - 图1](/uploads/projects/2book@sjms/d8de9a8cb5988601dbf16c2d72fcb6f2.png)
下面我们使用手机生产来讲解该模式:
Phone类:手机标准规范类(AbstractProduct)
public interface Phone {void make();}
MiPhone类:制造小米手机(Product1)
public class MiPhone implements Phone {public MiPhone() {this.make();}@Overridepublic void make() {// TODO Auto-generated method stubSystem.out.println("make xiaomi phone!");}}
IPhone类:制造苹果手机(Product2)
public class IPhone implements Phone {public IPhone() {this.make();}@Overridepublic void make() {// TODO Auto-generated method stubSystem.out.println("make iphone!");}}
PhoneFactory类:手机代工厂(Factory)
public class PhoneFactory {public Phone makePhone(String phoneType) {if(phoneType.equalsIgnoreCase("MiPhone")){return new MiPhone();}else if(phoneType.equalsIgnoreCase("iPhone")) {return new IPhone();}return null;}}
演示:
public class Demo {public static void main(String[] arg) {PhoneFactory factory = new PhoneFactory();Phone miPhone = factory.makePhone("MiPhone"); // make xiaomi phone!IPhone iPhone = (IPhone)factory.makePhone("iPhone"); // make iphone!}}
2. 工厂方法模式(Factory Method)
和简单工厂模式中工厂负责生产所有产品相比,工厂方法模式将生成具体产品的任务分发给具体的产品工厂,其UML类图如下:![[设计模式] 工厂模式 - 图2](/uploads/projects/2book@sjms/0065872f5d7760cdad784464a05eaf28.png)
也就是定义一个抽象工厂,其定义了产品的生产接口,但不负责具体的产品,将生产任务交给不同的派生类工厂。这样不用通过指定类型来创建对象了。
接下来继续使用生产手机的例子来讲解该模式。
其中和产品相关的Phone类、MiPhone类和IPhone类的定义不变。
AbstractFactory类:生产不同产品的工厂的抽象类
public interface AbstractFactory {Phone makePhone();}
XiaoMiFactory类:生产小米手机的工厂(ConcreteFactory1)
public class XiaoMiFactory implements AbstractFactory{@Overridepublic Phone makePhone() {return new MiPhone();}}
AppleFactory类:生产苹果手机的工厂(ConcreteFactory2)
public class AppleFactory implements AbstractFactory {@Overridepublic Phone makePhone() {return new IPhone();}}
演示:
public class Demo {public static void main(String[] arg) {AbstractFactory miFactory = new XiaoMiFactory();AbstractFactory appleFactory = new AppleFactory();miFactory.makePhone(); // make xiaomi phone!appleFactory.makePhone(); // make iphone!}}
3. 抽象工厂模式(Abstract Factory)
上面两种模式不管工厂怎么拆分抽象,都只是针对一类产品Phone(AbstractProduct),如果要生成另一种产品PC,应该怎么表示呢?
最简单的方式是把2中介绍的工厂方法模式完全复制一份,不过这次生产的是PC。但同时也就意味着我们要完全复制和修改Phone生产管理的所有代码,显然这是一个笨办法,并不利于扩展和维护。
抽象工厂模式通过在AbstarctFactory中增加创建产品的接口,并在具体子工厂中实现新加产品的创建,当然前提是子工厂支持生产该产品。否则继承的这个接口可以什么也不干。
其UML类图如下:![[设计模式] 工厂模式 - 图3](/uploads/projects/2book@sjms/76573f05dd25676a1b7500affe602ad7.png)
从上面类图结构中可以清楚的看到如何在工厂方法模式中通过增加新产品接口来实现产品的增加的。
接下来我们继续通过小米和苹果产品生产的例子来解释该模式。
为了弄清楚上面的结构,我们使用具体的产品和工厂来表示上面的UML类图,能更加清晰的看出模式是如何演变的:![[设计模式] 工厂模式 - 图4](/uploads/projects/2book@sjms/845bebdf33f1c52810a589568977b662.png)
PC类:定义PC产品的接口(AbstractPC)
public interface PC {void make();}
MiPC类:定义小米电脑产品(MIPC)
public class MiPC implements PC {public MiPC() {this.make();}@Overridepublic void make() {// TODO Auto-generated method stubSystem.out.println("make xiaomi PC!");}}
MAC类:定义苹果电脑产品(MAC)
public class MAC implements PC {public MAC() {this.make();}@Overridepublic void make() {// TODO Auto-generated method stubSystem.out.println("make MAC!");}}
下面需要修改工厂相关的类的定义:
AbstractFactory类:增加PC产品制造接口
public interface AbstractFactory {Phone makePhone();PC makePC();}
XiaoMiFactory类:增加小米PC的制造(ConcreteFactory1)
public class XiaoMiFactory implements AbstractFactory{@Overridepublic Phone makePhone() {return new MiPhone();}@Overridepublic PC makePC() {return new MiPC();}}
AppleFactory类:增加苹果PC的制造(ConcreteFactory2)
public class AppleFactory implements AbstractFactory {@Overridepublic Phone makePhone() {return new IPhone();}@Overridepublic PC makePC() {return new MAC();}}
演示:
public class Demo {public static void main(String[] arg) {AbstractFactory miFactory = new XiaoMiFactory();AbstractFactory appleFactory = new AppleFactory();miFactory.makePhone(); // make xiaomi phone!miFactory.makePC(); // make xiaomi PC!appleFactory.makePhone(); // make iphone!appleFactory.makePC(); // make MAC!}}
4.利用反射改造简单工厂模式
增加phone-config.properties配置文件:
MiPhone=com.phone.MiPhoneIPhone=com.phone.Iphone
增加反射工具类读取配置文件,并提供创建对象的方法:
package util;import java.io.File;import java.io.FileInputStream;import java.util.Properties;public class MyUtil {private static Properties properties=null;static{properties=new Properties();File file = new File("/Users/Mac/Java/demo/src/main/java/util/phone-config.properties");try{properties.load(new FileInputStream(file));}catch (Exception e) {throw new RuntimeException("配置文件加载失败....");}}public static Object getObject(String name) throws Exception{if(name == null || name.isEmpty()){throw new NullPointerException("name must have a value");}//第一步:获取类路径String classpath=properties.getProperty(name);//第二步:通过反射创建对象Class classObj=Class.forName(classpath);return classObj.newInstance();}}
改造后的工厂类代码:
package com.phone.factory;import com.phone.Phone;import util.MyUtil;public class PhoneFactory {public Phone makePhone(String phoneType){try {return (Phone) MyUtil.getObject(phoneType);} catch (Exception e) {e.printStackTrace();}return null;}}
演示代码:
package com;import com.phone.Phone;import com.phone.factory.PhoneFactory;public class Demo {public static void main(String[] arg) {PhoneFactory factory = new PhoneFactory();Phone miPhone = factory.makePhone("MiPhone");Phone iPhone = factory.makePhone("IPhone");/*运行结果:make xiaomi phone!make iphone!*/}}
5.总结
上面介绍的三种工厂模式有各自的应用场景,实际应用时能解决问题满足需求即可,可灵活变通,无所谓高级与低级。
此外无论哪种模式,由于可能封装了大量对象和工厂创建,新加产品需要修改已定义好的工厂相关的类,因此对于产品和工厂的扩展不太友好,利弊需要权衡一下。
参考资料: 设计模式之工厂模式
