哈喽,大家好,我是一条。 【白话设计模式】导航篇已更新,今天详细讲解第一个——工厂方法模式
目录
工厂方法模式是一种创建型设计模式, 其在父类中提供一个创建对象的方法, 允许子类决定实例化对象的类型。
假设你正在开发一款物流管理应用。最初版本只能处理卡车运输, 因此大部分代码都在位于名为 卡车
的类中。
一段时间后, 这款应用变得极受欢迎。你每天都能收到十几次来自海运公司的请求, 希望应用能够支持海上物流功能。
如果代码其余部分与现有类已经存在耦合关系, 那么向程序中添加新类其实并没有那么容易。
代码问题该如何处理呢?目前, 大部分代码都与 卡车
类相关。在程序中添加 轮船
类需要修改全部代码。更糟糕的是, 如果你以后需要在程序中支持另外一种运输方式, 很可能需要再次对这些代码进行大幅修改。
最后, 你将不得不编写繁复的代码, 根据不同的运输对象类, 在应用中进行不同的处理。
工厂方法模式建议使用特殊的工厂方法代替对于对象构造函数的直接调用 (即使用 new
运算符)。不用担心, 对象仍将通过 new
运算符创建, 只是该运算符改在工厂方法中调用罢了。工厂方法返回的对象通常被称作 “产品”。
子类可以修改工厂方法返回的对象类型。
乍看之下, 这种更改可能毫无意义:我们只是改变了程序中调用构造函数的位置而已。但是, 仔细想一下, 现在你可以在子类中重写
工厂方法, 从而改变其创建产品的类型。
但有一点需要注意:仅当这些产品具有共同的基类或者接口时, 子类才能返回不同类型的产品, 同时基类中的工厂方法还应将其返回类型声明为这一共有接口。
所有产品都必须使用同一接口。
举例来说, 卡车
Truck和 轮船
Ship类都必须实现 运输
Transport接口, 该接口声明了一个名为 deliver
的方法。
每个类都将以不同的方式实现该方法:卡车走陆路交付货物, 轮船走海路交付货物。 陆路运输
RoadLogistics类中的工厂方法返回卡车对象, 而 海路运输
SeaLogistics类则返回轮船对象。
只要产品类实现一个共同的接口, 你就可以将其对象传递给客户代码, 而无需提供额外数据。
调用工厂方法的代码 (通常被称为客户端代码) 无需了解不同子类返回实际对象之间的差别。客户端将所有产品视为抽象的 运输
。客户端知道所有运输对象都提供 交付
方法, 但是并不关心其具体实现方式。
package com.yitiao.demo.factoryMethod;
/**
* @Author: 一条IT
* @Date: 2021/6/8 22:46
*/
public interface Transport {
void deliver();
}
package com.yitiao.demo.factoryMethod;
/**
* @Author: 一条IT
* @Date: 2021/6/8 23:00
*/
public class Ship implements Transport {
@Override
public void deliver() {
System.out.println("Ship");
}
}
package com.yitiao.demo.factoryMethod;
/**
* @Author: 一条IT
* @Date: 2021/6/8 23:00
*/
public class Truck implements Transport {
@Override
public void deliver() {
System.out.println("Truck");
}
}
package com.yitiao.demo.factoryMethod;
/**
* @Author: 一条IT
* @Date: 2021/6/8 23:07
*/
public abstract class Logistics {
public abstract Transport createTransprot();
}
package com.yitiao.demo.factoryMethod;
/**
* @Author: 一条IT
* @Date: 2021/6/8 23:13
*/
public class RoadLogistics extends Logistics{
@Override
public Transport createTransprot() {
return new Ship();
}
}
package com.yitiao.demo.factoryMethod;
/**
* @Author: 一条IT
* @Date: 2021/6/8 23:14
*/
public class SeaLogistics extends Logistics {
@Override
public Transport createTransprot() {
return new Truck();
}
}
package com.yitiao.demo.factoryMethod;
/**
* @Author: 一条IT
* @Date: 2021/6/8 23:15
*/
public class Test {
public static void main(String[] args) {
Logistics roadLogistics = new RoadLogistics();
roadLogistics.createTransprot().deliver();
Logistics seaLogistics = new SeaLogistics();
seaLogistics.createTransprot().deliver();
}
}
在这里插入图片描述
复杂度: 0.3
流行度: 1
使用示例: 工厂方法模式在 Java 代码中得到了广泛使用。当你需要在代码中提供高层次的灵活性时, 该模式会非常实用。
核心 Java 程序库中有该模式的应用:
java.util.Calendar#getInstance()
java.util.ResourceBundle#getBundle()
java.text.NumberFormat#getInstance()
java.nio.charset.Charset#forName()
java.net.URLStreamHandlerFactory#createURLStreamHandler(String)
(根据协议返回不同的单例对象)java.util.EnumSet#of()
javax.xml.bind.JAXBContext#createMarshaller()
及其他类似的方法。识别方法: 工厂方法可通过构建方法来识别, 它会创建具体类的对象, 但以抽象类型或接口的形式返回这些对象。
无法预知对象确切类别及其依赖关系时
工厂方法将创建产品的代码与实际使用产品的代码分离, 从而能在不影响其他代码的情况下扩展产品创建部分代码。
例如, 如果需要向应用中添加一种新产品, 你只需要开发新的创建者子类, 然后重写其工厂方法即可。
希望用户能扩展你软件库或框架的内部组件
继承可能是扩展软件库或框架默认行为的最简单方法。但是当你使用子类替代标准组件时, 框架如何辨识出该子类?
解决方案是将各框架中构造组件的代码集中到单个工厂方法中, 并在继承该组件之外允许任何人对该方法进行重写。
让我们看看具体是如何实现的。假设你使用开源 UI 框架编写自己的应用。你希望在应用中使用圆形按钮, 但是原框架仅支持矩形按钮。你可以使用 圆形按钮
RoundButton子类来继承标准的 按钮
Button类。但是, 你需要告诉 UI框架
UIFramework类使用新的子类按钮代替默认按钮。
为了实现这个功能, 你可以根据基础框架类开发子类 圆形按钮 UI
UIWithRoundButtons , 并且重写其 createButton
创建按钮方法。基类中的该方法返回 按钮
对象, 而你开发的子类返回 圆形按钮
对象。现在, 你就可以使用 圆形按钮 UI
类代替 UI框架
类。就是这么简单!
希望复用现有对象来节省系统资源
在处理大型资源密集型对象 (比如数据库连接、 文件系统和网络资源) 时, 你会经常碰到这种资源需求。
让我们思考复用现有对象的方法:
这些代码可不少!而且它们必须位于同一处, 这样才能确保重复代码不会污染程序。
可能最显而易见, 也是最方便的方式, 就是将这些代码放置在我们试图重用的对象类的构造函数中。但是从定义上来讲, 构造函数始终返回的是新对象, 其无法返回现有实例。
因此, 你需要有一个既能够创建新对象, 又可以重用现有对象的普通方法。这听上去和工厂方法非常相像。
switch
分支运算符, 用于选择各种需要实例化的产品类。但是不要担心, 我们很快就会修复这个问题。邮件
及其子类 航空邮件
和 陆路邮件
; 运输
及其子类 飞机
, 卡车
和 火车
。 航空邮件
仅使用 飞机
对象, 而 陆路邮件
则会同时使用 卡车
和 火车
对象。你可以编写一个新的子类 (例如 火车邮件
) 来处理这两种情况, 但是还有其他可选的方案。客户端代码可以给 陆路邮件
类传递一个参数, 用于控制其希望获得的产品。我是一条,一个在互联网摸爬滚打的程序员。
道阻且长,行则将至。大家的 【点赞,收藏,关注】 就是一条创作的最大动力,我们下期见!
注:关于本篇博客有任何问题和建议,欢迎大家留言!