2.2 解 决 方 案

2.2.1 使用简单工厂来解决问题

用来解决上述问题的一个合理的解决方案就是简单工厂,那么什么是简单工厂呢?

1.简单工厂的定义

提供一个创建对象实例的功能,而无须关心其具体实现。被创建实例的类型可以是接口、抽象类,也可以是具体的类。

2.应用简单工厂来解决问题的思路

分析上面的问题,虽然不能让模块外部知道模块内部的具体实现,但是模块内部是可以知道实现类的,而且创建接口是需要具体实现类的。

那么,干脆在模块内部新建一个类,在这个类里面来创建接口,然后把创建好的接口返回给客户端,这样,外部应用就只需要根据这个类来获取相应的接口对象,然后就可以操作接口定义的方法了。把这样的对象称为简单工厂,就叫它Factory吧。

这样一来,客户端就可以通过Factory来获取需要的接口对象,然后调用接口的方法来实现需要的功能,而且客户端也不用再关心具体的实现了。

2.2.2 简单工厂的结构和说明

简单工厂的结构如图2.5所示。

图片

图2.5 简单工厂的结构示意图

■ Api:定义客户所需要的功能接口。

■ Impl:具体实现Api的实现类,可能会有多个。

■ Factory:工厂,选择合适的实现类来创建Api接口对象。

■ Client:客户端,通过Factory来获取Api接口对象,然后面向Api接口编程。

2.2.3 简单工厂示例代码

(1)Api定义的示例代码如下:

c745a385488d4cd4ab7cdc52cbbcdc6b

(2)定义了接口,接下来实现它。ImplA的示例代码如下:

4ea885757011409f9c09d108496c861e

ImplB的示意实现和ImplA基本一样。示例代码如下:

c30d70cc06734ed4ae4263c2640af067

(3)下面来看看简单工厂的实现。示例代码如下:

f727e77e6f7c4b4682f3d2e3ac6523f2

(4)再来看看客户端的示意,示例代码如下:

1c1dec15acf8415c86219eaa6504c101

096502f6c637464d8ca256a3cd9b1c76

2.2.4 使用简单工厂重写示例

要使用简单工厂来重写前面的示例,主要就是要创建一个简单工厂对象,让简单工厂来负责创建接口对象。然后让客户端通过工厂来获取接口对象,而不再由客户端自己去创建接口的对象了。

此时系统的结构如图2.6所示。

图片

图2.6 使用简单工厂重写示例的结构示意图

(1)接口Api和实现类Impl都和前面的示例一样,这里不再赘述。

(2)新创建一个简单工厂的对象。示例代码如下:

9b7a790763684aa2acaae868cf611da7

ce029ef0a0d3487c854e59f683c93452

(3)使用简单工厂。

客户端如何使用简单工厂提供的功能呢?这个时候,客户端就不用再自己去创建接口的对象了,应该使用工厂来获取,经过改造,客户端代码如下:

ec3d5fd1e90243629947e10c225a5652

就如同上面的示例,客户端通过简单工厂创建了一个实现接口的对象,然后面向接口编程,从客户端来看,它根本不知道具体的实现是什么,也不知道是如何实现的,它只知道通过工厂获得了一个接口对象,然后通过这个接口来获取想要的功能。

事实上,简单工厂能帮助我们真正地开始面向接口编程,像以前的做法,其实只是用到了接口的多态部分的功能,而最重要的“封装隔离性”并没有体现出来。