Java设计模式-工厂方法

定义一个用于创建对象的接口,Factory Method将一个类的实例化延迟到了子类。

1、介绍

工厂方法模式的功能

工厂方法的主要功能是让父类在不知道具体实现的情况下,完成自身的功能调用,而具体的实现延迟到子类来实现。

实现成抽象类

工厂方法的实现中,通常父类会是一个抽象类,里面包含创建所需对象的抽象方法,这些抽象方法就是工厂方法

实现成具体的类

也可以把父类实现成为一个具体的类,这种情况下,通常是在父类中提供获取所需对象的默认实现方法,这样就算没有具体的子类,也能够运行。

工厂方法的参数和返回值

工厂方法的实现中,可能需要参数,以便决定到底选用哪一种具体的实现。一般工厂方法返回的是被创建对象的接口对象,当然也可以是抽象类或者一个具体的类的实例。

谁来使用工厂方法创建的对象

(1)在工厂方法模式里面,应该是Creator中的其它方法在使用工厂方法创建的对象

(2)客户端应该是使用Creator对象,或者是使用由Creator创建出来的对象,这个时候工厂方法创建的对象,是Creator中的某些方法使用。

(3)在某些情况下,客户端可能会使用由Creator创建出来的对象,这个时候工厂方法创建的对象,是构成客户端需要的对象的一部分。

工厂方法模式的本质

延迟到子类来选择实现

对设计原则的体现

工厂方法模式很好的体现了“依赖倒置原则” 。依赖倒置原则告诉我们“要依赖抽象,不要依赖于具体类”,简单点说就是:不能让高层组件依赖于低层组件,而且不管高层组件还是低层组件,都应该依赖于抽象。

何时选用工厂方法模式

如果一个类需要创建某个接口的对象,但是又不知道具体的实现,这种情况可以选用工厂方法模式,把创建对象的工作延迟到子类去实现

如果一个类本身就希望,由它的子类来创建所需的对象的时候,应该使用工厂方法模式

工厂方法模式的调用顺序示意图

客户端使用由Creator创建出来的对象情况的调用顺序示意图

客户端使用Creator对象时候的调用顺序示意图

2、结构说明

Product:

定义工厂方法所创建的对象的接口,也就是实际需要使用的对象的接口

ConcreteProduct:

具体的Product接口的实现对象。

Creator:

创建器,声明工厂方法

ConcreteCreator:

具体的创建器对象,覆盖实现Creator定义的工厂方法,返回具体的Product实例

3、工厂方法模式优缺点

可以在不知具体实现的情况下编程

更容易扩展对象的新版本

连接平行的类层次

具体产品对象和工厂方法的耦合性

4、理解工厂方法模式

工厂方法模式与IoC/DI

(1)概念

依赖注入:应用程序依赖容器创建并注入它所需要的外部资源

控制反转:容器控制应用程序,由容器反向的向应用程序注入应用程序所需要的外部资源。

(2)过程

没有IoC/DI的时候,常规的A类使用C类的示意图

有了IoC/DI的容器后,A类不再主动去创建C了

而是被动等待,等待IoC/DI的容器获取一个C的实例,然后反向的注入到A类中

(3)思想

其实IoC/DI对编程带来的最大改变不是从代码上,而是从思想上,发生了“主从换位”的变化。应用程序原本是老大,要获取什么资源都是主动出击,但是在IoC/DI思想中,应用程序就变成被动的了,被动的等待IoC/DI容器来创建并注入它所需要的资源了。

这么小小的一个改变其实是编程思想的一个大进步,这样就有效的分离了对象和它所需要的外部资源,使得它们松散耦合,有利于功能复用,更重要的是使得程序的整个体系结构变得非常灵活。

(4)工厂方法模式和IoC/DI的关系

他们的思想很类似,都是“主动变被动”,进行“主从换位”,从而获得更灵活的程序结构

平行的类层次结构

(1)什么是平行的类层次结构呢?

简单点说,假如有两个类层次结构,其中一个类层次中的每个类在另一个类层次中都有一个对应的类的结构,就被称为平行的类层次结构。

(2)这种平行的类层次结构用来干什么呢?

主要用来把一个类层次中的某些行为分离出来,让类层次中的类把原本属于自己的职责,委托给分离出来的类去实现,从而使得类层次本身变得更简单,更容易扩展和复用。

(3)工厂方法模式跟平行的类层次结构有何关系呢?

可以使用工厂方法模式来连接平行的类层次。

参数化工厂方法

所谓参数化工厂方法指的就是:通过给工厂方法传递参数,让工厂方法根据参数的不同来创建不同的产品对象。

5、代码示例

/**
 * 导出的文件对象的接口
 */
public interface ExportFileApi {
	/**
	 * 导出内容成为文件
	 * @param data 示意:需要保存的数据
	 * @return 是否导出成功
	 */
	public boolean export(String data);
}

/**
 * 实现导出数据的业务功能对象
 */
public abstract class ExportOperate {
	
	
	/**
	 * 导出文件
	 * @param data 需要保存的数据
	 * @return 是否成功导出文件
	 */
	public boolean export(String data){
		//比如进行数据校验
		//比如进行数据转换
		//比如进行数据格式的封装
		
		//使用工厂方法导出
		ExportFileApi api = factoryMethod();
		
		return api.export(data);
	}
	/**
	 * 工厂方法,创建导出的文件对象的接口对象
	 * @return 导出的文件对象的接口对象
	 */
	protected abstract ExportFileApi factoryMethod();
	
	//既要约束子类的行为,又要为子类提供公共的功能
}

/**
 * 导出成数据库备份文件形式的对象
 */
public class ExportDB implements ExportFileApi{
	public boolean export(String data) {
		//简单示意一下,这里需要操作数据库和文件
		System.out.println("导出数据"+data+"到数据库备份文件");
		return true;
	}
}

/**
 * 具体的创建器实现对象,实现创建导出成数据库备份文件形式的对象
 */
public class ExportDBOperate extends ExportOperate{
	protected ExportFileApi factoryMethod() {
		//创建导出成数据库备份文件形式的对象
		return new ExportDB();
	}
}

/**
 * 导出成文本文件格式的对象
 */
public class ExportXmlFile implements ExportFileApi{
	public boolean export(String data) {
		//简单示意一下,这里需要操作文件
		System.out.println("导出数据"+data+"到xml文件");
		return true;
	}
}

/**
 * 导出成文本文件格式的对象
 */
public class ExportXmlFileOperate extends ExportOperate{

	@Override
	protected ExportFileApi factoryMethod() {
		return new ExportXmlFile();
	}	
}

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值