超级链接: Java常用设计模式的实例学习系列-绪论
参考:《HeadFirst设计模式》
工厂存在的意义是:高效的制造某一类产品。
工厂模式存在的意义是:高效的创建某一类对象。
工厂模式有很多种,本文只介绍以下三种:
简单工厂模式
,又称静态工厂模式
。工厂方法模式
,全称多个工厂方法模式
。抽象工厂模式
。本文以`手机制造工厂生产手机为场景来学习工厂模式。
关于手机的编码不过多的叙述,就是很简单的一个父抽象类加两个子实现类的关系。
父抽象手机类:Phone
/**
* 手机
*
* @author hanchao
*/
@Getter
@Setter
@AllArgsConstructor
public abstract class Phone {
private String name;
/**
* 打个电话测试一下
*/
public abstract void callTest();
}
子实现类:苹果手机:ApplePhone
/**
* 苹果手机
*
* @author hanchao
*/
@Slf4j
public class ApplePhone extends Phone {
public ApplePhone() {
super("苹果手机");
}
@Override
public void callTest() {
log.info("使用[" + super.getName() + "]给13322224444打电话.");
}
}
子实现类:安卓手机:AndroidPhone
/**
* 安卓手机
*
* @author hanchao
*/
@Slf4j
public class AndroidPhone extends Phone {
public AndroidPhone() {
super("安卓手机");
}
@Override
public void callTest() {
log.info("使用[" + super.getName() + "]给15533335555打电话.");
}
}
简单工厂模式:一般情况下,创建对象的方法需要传入代表类型的参数type,然后再通过if-else或者switch语句根据type创建对应的对象。
直接上代码:PhoneFactory
/**
* 简单工厂:手机工厂
*
* @author hanchao
*/
public class PhoneFactory {
public static final int TYPE_APPLE = 1;
public static final int TYPE_ANDROID = 2;
/**
* 简单工厂:一般情况下,创建对象的方法需要传入代表类型的参数type,然后再通过if-else或者switch语句根据type创建对应的对象。
*/
public static Phone producePhone(int type) {
Phone phone;
switch (type) {
case TYPE_APPLE:
//制造一个苹果手机
phone = new ApplePhone();
break;
case TYPE_ANDROID:
//制造一个安卓手机
phone = new AndroidPhone();
break;
default:
throw new IllegalArgumentException("此型号手机暂不制造!");
}
return phone;
}
}
简单工厂的机制很简单:根据传入的类型,创建对应的对象。
这里将可用的类型编写成了静态变量,这用能在一定程度上减少出错的可能。
然后,写一个简单的测试类:SimpleFactoryDemo
/**
* 1.简单工厂模式
*
* @author hanchao
*/
public class SimpleFactoryDemo {
public static void main(String[] args) {
//某客户下了一笔大单子,要求制造2台苹果手机和1台安卓手机
//于是手机工厂开始了制造
//先来一台苹果手机
Phone phone = PhoneFactory.producePhone(PhoneFactory.TYPE_APPLE);
//测试一下
phone.callTest();
//再来一台苹果手机
phone = PhoneFactory.producePhone(PhoneFactory.TYPE_APPLE);
//测试一下
phone.callTest();
//再来一台安卓手机
phone = PhoneFactory.producePhone(PhoneFactory.TYPE_ANDROID);
//测试一下
phone.callTest();
}
}
运行结果:
2019-06-30 15:03:10,353 INFO [main-1] pers.hanchao.designpattern.factory.simple.telephone.ApplePhone:19 - 使用[苹果手机]给13322224444打电话.
2019-06-30 15:03:10,357 INFO [main-1] pers.hanchao.designpattern.factory.simple.telephone.ApplePhone:19 - 使用[苹果手机]给13322224444打电话.
2019-06-30 15:03:10,358 INFO [main-1] pers.hanchao.designpattern.factory.simple.telephone.AndroidPhone:19 - 使用[安卓手机]给15533335555打电话.
**简单工厂:**一般情况下,创建对象的方法需要传入代表类型的参数type,然后再通过if-else或者switch语句根据type创建对应的对象。
通常提供的创建对象的方法是静态的
,所以简单工厂模式
也称之为静态工厂模式
。
实现简单
:代码逻辑很简单。可复用高
:所有对象的创建都通过同名的方法创建。容错率低
,如果传入的类型参数type因为粗心写错,则创建的产品会与预期不符。耦合度高
,每增减一种产品,都需要对创建对象的唯一方法进行修改。扩展性差
,每增减一种产品,都需要修改工厂类,违背了设计模式的开放-封闭原则(OCP,对扩展开放,对修改关闭)。造成以上缺点的关键在于:所有对象的创建都通过唯一的工厂方法。
为了解决以上问题,尝试为每种对象提供相应的工厂方法
。如此进化成了:多个工厂方法模式。
(多个)工厂方法模式的关键点在于:为每种对象提供相应的工厂方法
,所以很容易得到如下的工厂类:
PhoneFactory
/**
* 多个工厂方法模式:手机工厂
*
* @author hanchao
*/
public class PhoneFactory {
/**
* 制造一个苹果手机
*/
public static Phone produceApplePhone() {
return new ApplePhone();
}
/**
* 制造一个安卓手机
*/
public static Phone produceAndroidPhone() {
return new AndroidPhone();
}
}
然后,写一个简单的测试类:MultiFactoryMethodsDemo
/**
* 2.多个工厂方法模式
*
* @author hanchao
*/
public class MultiFactoryMethodsDemo {
public static void main(String[] args) {
//某客户下了一笔大单子,要求制造2台苹果手机和1台安卓手机
//于是手机工厂开始了制造
//先来一台苹果手机
Phone phone = PhoneFactory.produceApplePhone();
//测试一下
phone.callTest();
//再来一台苹果手机
phone = PhoneFactory.produceApplePhone();
//测试一下
phone.callTest();
//再来一台安卓手机
phone = PhoneFactory.produceAndroidPhone();
//测试一下
phone.callTest();
}
}
运行结果:略
多个工厂方法模式
: 每种对象分别编写专用的创建方法,根据调用的创建方法的不同,产生不同的对象。
多个工厂方法模式
又简称为工厂方法模式
。
实现简单
:代码逻辑很简单。容错率高
,因为每种对象都有专用的创建方法,无需传入类型等参数,只要调用正确的创建方法就不会出错。耦合度低
,每增减一种产品,只需要增减一个创建方法,不会影响其他对象的创建方法。扩展性差
,每增减一种产品,都需要修改工厂类,违背了设计模式的开放-封闭原则(OCP,对扩展开放,对修改关闭)。可复用低
,所有对象的创建方法都不同名。造成以上缺点的关键在于:所有对象的创建方法都存在于唯一个工厂。
为了解决以上问题,尝试为每种对象提供单独的工厂及工厂方法
。如此进化成了:抽象工厂模式。
抽象工厂模式
的关键点在于:``为每种对象提供单独的工厂及工厂方法`。
所以需要把工厂类抽象出来,形成抽象工厂,然后针对每种对象分别实现专用的工厂及工厂方法,这就是命名抽象工厂
的由来。
抽象工厂:AbstractPhoneFactory
/**
* 抽象手机工厂
*
* @author hanhao
*/
public abstract class AbstractPhoneFactory {
/**
* 制造手机
*
* @return 手机
*/
public abstract Phone producePhone();
}
苹果手机专用的工厂及创建方法:ApplePhoneFactory
/**
* 具体工厂:苹果手机工厂
*
* @author hanchao
*/
public class ApplePhoneFactory extends AbstractPhoneFactory {
@Override
public Phone producePhone() {
return new ApplePhone();
}
}
安卓手机专用的工厂及创建方法:AndroidPhoneFactory
/**
* 具体工厂:安卓手机工厂
*
* @author hanchao
*/
public class AndroidPhoneFactory extends AbstractPhoneFactory {
@Override
public Phone producePhone() {
return new AndroidPhone();
}
}
然后,写一个简单的测试类:AbstractFactoryDemo
/**
* 3.抽象工厂模式
*
* @author hanchao
*/
public class AbstractFactoryDemo {
public static void main(String[] args) {
//某客户下了一笔大单子,要求制造2台苹果手机和1台安卓手机
//于是手机工厂开始了制造
//先来一台苹果手机
Phone phone = new ApplePhoneFactory().producePhone();
//测试一下
phone.callTest();
//再来一台苹果手机
phone = new ApplePhoneFactory().producePhone();
//测试一下
phone.callTest();
//再来一台安卓手机
phone = new AndroidPhoneFactory().producePhone();
//测试一下
phone.callTest();
}
}
运行结果:略
抽象工厂模式
: 定义一个抽象工厂和抽象方法,每种对象都有专用的子工厂及对象创建方法。
容错率高
,因为每种对象都有专用的工厂及创建方法,只要调用正确的工厂的创建方法就不会出错。耦合度低
,每增减一种产品,只需增减一个工厂及创建方法,不会影响其他对象的创建方法。扩展性好
,每增减一种产品,只需增减此对象的工厂及创建方法,不修改其他工厂,遵循了开放-封闭原则(OCP,对扩展开放,对修改关闭)。可复用高
,所有对象的创建都通过同名的方法创建。实现复杂
:代码逻辑相对复杂。可能100种对象,需要定义100个工厂类。最后以UML类图来总结本文的三种工厂模式。
从设计模式原则考虑,上述三种工厂模式的推荐优先级:抽象工厂模式
>>>(多个)工厂方法模式
>>>简单/静态工厂模式
。
不过凡事不绝对,如果你说你能百分百的保证业务场景不会发生变动,那你是用哪种模式都是无所谓的。