厂方法模式
工厂方法模式是类的创建模式。工厂方法模式的用意是定义一个创建产品对象的工厂接口,将实际创建工厂推迟到子类中。
工厂方法模式
工厂方法模式是对简单工厂模式进一步抽象的结果。
假如是不使用反射的工厂方法模式,那么所有的if...
else if...else都放在工厂类中,势必造成工厂类的无限臃肿
这时候就需要工厂方法模式来处理这个问题了。工厂方法模式中,核心的工厂类不再负责所有对象的创建,而是将具体的创建工作交给子类去做。这个类则摇身一变变成了一个抽象工厂角色,仅仅负责给出具体工厂子类必须实现的接口。
这一步的改进,使得系统可以在不修改具体工厂角色的情况下引入新的产品。
工厂方法模式结构
使用工厂方法模式的系统涉及到以下角色:
1、抽象工厂角色
担任这个角色的是工厂方法模式的核心,任何在模式中创建对象的工厂类必须实现这个接口
2、具体工厂角色
担任这个角色的是实现了工厂接口的具体Java类,具体工厂角色与业务密切相关,并且随着使用者的调用以创建导出类
3、抽象导出角色
工厂方法模式所创建的对象的超类
4、具体导出角色
这个角色实现了抽象导出角色所声明的接口,工厂方法模式所创建的每一个对象都是某个具体角色导出角色的实例
工厂方法模式的示例
有一个抽象工厂,导出文件的工厂:
它有两个实现类,分别是导出HTML文件的工厂和导出PDF文件的工厂:
抽象产品角色,一个导出文件:
具体产品角色:
模拟客户端调用,实例化出一个具体的工厂角色,根据传入的参数返回不同的产品角色:
返回结果为:
导出财务版HTML文件
当然,也可以修改代码,看自己喜好,实例化出不同的子类并且调用export方法打印。
工厂方法模式在Java中的应用及解读
拿ThreadFactory举个例子,顾名思义,这是一个生产线程的接口:
具体的线程工厂可以implements这个接口并实现newThread(Runnable r)方法,来生产具体线程工厂想要生产的线程。JDK在Executors给开发者提供了一个静态内部类DefaultThreadFactory,当然开发者也可以自行实现这个接口,写自定义的线程工厂。
总结
对于系统的设计应该足够灵活并尽可能降低代码之间的耦合度,当修改或增加一个新的功能时,使得使用者尽可能修改少的地方即可。假如设计不够灵活,将无法面对多变的需求,可能一个极小的需求变更,都会使代码结构发生改变,并导致其他使用的人都要修改他们的代码。牵一发而动全身,系统日后的维护将变得艰难。
ava设计模式
本系列文章共23篇,详细介绍GOF (Gang Of Four)所定义的23种设计模式。共分为三大类对应标题中的3大招,每类中的每一种设计模式对应3大招中的某一式:
第1招-创建型(共5式):单例模式,工厂方法模式,抽象工厂模式,建造者模式,原型模式;
第2招-行为型(共11式):观察者模式,策略模式,命令模式,状态模式,解释器模式,迭代器模式,中介者模式,访问者模式,备忘录模式,责任链模式,模板方法模式;
第3招-结构型(共7式):适配器模式,外观模式,代理模式,装饰者模式,组合模式,桥接模式,享元模式。
UML (Unified Modeling Language, 统一建模语言):9大图——构件图、类图、对象图、序列图、协作图、活动图、状态机图、用例图、部署图;6关系——泛化=实现>组合>聚合>关联>依赖。
OO(Object-Oriented, 面向对象):面向对象的程序设计方法将数据及对数据的操作封装在一起形成一个相互依赖不可分离的整体,即对象;对同类型的对象抽象出其共性形成类。
定义
应用场景
优缺点
Java代码实例
小结
工厂方法模式:对象的工厂,将类的实例化过程延迟到子类进行。
说到工厂模式就不得不提一下工厂模式三姐妹,简单工厂模式、工厂方法模式和抽象工厂模式。其中简单工厂模式存在诸多违背设计原则的问题,未被列入23种设计模式。而工厂方法模式解决了这些问题,特别是“开放关闭原则”,提供了工厂类的抽象,实现了可扩展性。抽象工厂模式不是本文的重点,留到下节介绍。
工厂方法模式涉及到四个角色:具体产品、抽象产品、具体工厂、抽象工厂。
工厂方法模式UML图
产品种类不确定,后期可能添加新的产品时;
当前类并不知道需要创建的是什么子类时;
不想在父类进行实例化工作,想让子类决定时。
优点:
良好的扩展性,当需要添加新的产品时无需修改原来的类,只需添加一个新的产品类和一个工厂类即可;
代码结构清晰,产品类是产品类,工厂负责创建,客户端无需知道各种细节,解耦了产品的创建过程;
增加了维护性,将创建过程放到了工厂类里,某一产品的创建发生变化时不会影响产品本身;
缺点:
类的个数爆发式增长,每添加一种产品就需要添加两个类,如果产品过多则导致类过多,增加类定义的内存开销;
在实现工厂方法模式这前我们先来看一下简单工厂模式:
1.创建抽象产品IPhone;
2.创建具体产品MiPhone和ApplePhone;
抽象产品和具体产品
3.创建简单工厂PhoneFactory;
4.使用工厂类进行创建产品;
简单工厂类及客户端类
实现工厂方法模式需要四步:
1.定义产品接口;
2.添加具体产品实现;
抽象产品及具体产品
3.定义创建工厂接口;
4.添加具体工厂实现;
抽象工厂和具体工厂
5.客户端使用。
客户端使用
工厂模式三姐妹中最常用的是简单工厂模式,但简单工厂模式违背了设计原则中的开放关闭原则。而工厂方法模式解决了此问题,并且完美使用了里氏替换原则和依赖倒置原则。对于可能出现多种类别的实例实现时,建议使用工厂方法模式进行解耦。
工厂模式(Factory Pattern),根据不同的名称输入返回不同类的实例,一般用来创建同一类对象。工厂模式的主要思想是将对象的创建与对象的实现分离。
在类似场景中,这些例子有以下特点:
工厂模式分为简单工厂模式、工厂方法模式、抽象工厂模式。
简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例
简单工厂模式包含如下角色:
类图: AmericanoCoffee、LatteCoffee和CappuccinoCoffee都是继承Coffee
代码
abstract class Coffee {
constructor(public name: string) {
}
}
class AmericanoCoffee extends Coffee {
constructor(public name: string) {
super(name);
}
}
class LatteCoffee extends Coffee {
constructor(public name: string) {
super(name);
}
}
class CappuccinoCoffee extends Coffee {
constructor(public name: string) {
super(name);
}
}
class Café {
static order(name: string) {
switch (name) {
case 'Americano':
return new AmericanoCoffee('美式咖啡');
case 'Latte':
return new LatteCoffee('拿铁咖啡');
case 'Cappuccino':
return new LatteCoffee('卡布奇诺');
default:
return null;
}
}
}
console.log(Café.order('Americano'));
console.log(Café.order('Latte'));
console.log(Café.order('Cappuccino'));
jQuery
class jQuery{
constructor(selector){
let elements = Array.from(document.querySelectorAll(selector));
let length = elements?elements.length:0;
for(let i=0;i<length;i++){
this[i]=elements[i];
}
this.length = length;
}
html(html){
if(html){
this[0].innerHTML=html;
}else{
return this[0].innerHTML;
}
}
}
window.$ = function(selector){
return new jQuery(selector);
}
和原生的 document.createElement 类似,Vue 和 React 这种具有虚拟 DOM 树(Virtual Dom Tree)机制的框架在生成虚拟 DOM 的时候,都提供了 createElement 方法用来生成 VNode,用来作为真实 DOM 节点的映射
// Vue
createElement('h3', { class: 'main-title' }, [
createElement('img', { class: 'avatar', attrs: { src: '../avatar.jpg' } }),
createElement('p', { class: 'user-desc' }, '放弃不难,但坚持一定很酷')
])
// React
React.createElement('h3', { className: 'user-info' },
React.createElement('img', { src: '../avatar.jpg', className: 'avatar' }),
React.createElement('p', { className: 'user-desc' }, '放弃不难,但坚持一定很酷')
)
createElement 函数结构大概如下:
class Vnode (tag, data, children) { ... }
function createElement(tag, data, children) {
return new Vnode(tag, data, children)
}
可以看到 createElement 函数内会进行 VNode 的具体创建,创建的过程是很复杂的,而框架提供的 createElement 工厂方法封装了复杂的创建与验证过程,对于使用者来说就很方便了。
工厂模式在源码中应用频繁,以 vue-router 中的源码为例,代码位置:vue-router/src/index.js
// src/index.js
export default class VueRouter {
constructor(options) {
this.mode = mode // 路由模式
switch (mode) { // 简单工厂
case 'history': // history 方式
this.history = new HTML5History(this, options.base)
break
case 'hash': // hash 方式
this.history = new HashHistory(this, options.base, this.fallback)
break
case 'abstract': // abstract 方式
this.history = new AbstractHistory(this, options.base)
break
default:
// ... 初始化失败报错
}
}
}
稍微解释一下这里的源码。mode 是路由创建的模式,这里有三种 History、Hash、Abstract,前两种我们已经很熟悉了,History 是 H5 的路由方式,Hash 是路由中带 # 的路由方式,Abstract 代表非浏览器环境中路由方式,比如 Node、weex 等;this.history 用来保存路由实例,vue-router 中使用了工厂模式的思想来获得响应路由控制类的实例。
源码里没有把工厂方法的产品创建流程封装出来,而是直接将产品实例的创建流程暴露在 VueRouter 的构造函数中,在被 new 的时候创建对应产品实例,相当于 VueRouter的构造函数就是一个工厂方法。
如果一个系统不是 SPA (Single Page Application,单页应用),而是是 MPA(Multi Page Application,多页应用),那么就需要创建多个 VueRouter 的实例,此时 VueRouter 的构造函数也就是工厂方法将会被多次执行,以分别获得不同实例。
简单工厂模式是根据输入的不同返回不同产品的实例;而工厂方式的主要思想是增加抽象工厂类,将不同产品的创建分离到不同的工厂中,工厂类的职责比较单一。
abstract class Coffee {
constructor(public name: string) {
}
}
abstract class Factory {
abstract createCoffee(): Coffee;
}
class AmericanoCoffee extends Coffee {
constructor(public name: string) {
super(name);
}
}
class AmericanoCoffeeFactory extends Factory {
createCoffee() {
return new AmericanoCoffee('美式咖啡')
}
}
class LatteCoffee extends Coffee {
constructor(public name: string) {
super(name);
}
}
class LatteCoffeeFactory extends Factory {
createCoffee() {
return new LatteCoffee('拿铁咖啡')
}
}
class CappuccinoCoffee extends Coffee {
constructor(public name: string) {
super(name);
}
}
class CappuccinoFactory extends Factory {
createCoffee() {
return new CappuccinoCoffee('卡布奇诺')
}
}
class Café {
static order(name: string) {
switch (name) {
case 'Americano':
return new AmericanoCoffeeFactory().createCoffee();
case 'Latte':
return new LatteCoffeeFactory().createCoffee();
case 'Cappuccino':
return new CappuccinoFactory().createCoffee();
default:
return null;
}
}
}
console.log(Café.order('Americano'));
console.log(Café.order('Latte'));
console.log(Café.order('Cappuccino'));
const settings={
'Americano': AmericanoCoffeeFactory,
'Latte': LatteCoffeeFactory,
'Cappuccino': CappuccinoFactory
}
console.log(new settings('Americano').createCoffee());
console.log(new settings('Latte').createCoffee());
console.log(new settings('Cappuccino').createCoffee());
带来了额外的系统复杂度,增加了抽象性。
那在什么时候使用工厂模式呢:
什么时候不该用工厂模式:
滥用只是增加了不必要的系统复杂度,过犹不及。
*请认真填写需求信息,我们会在24小时内与您取得联系。