三言两语说透设计模式的艺术-抽象工厂模式

开发 前端
抽象工厂模式(Abstract Factory Pattern)是一种软件设计模式,它属于创建型模式,其主要目的是创建一组相关或互相依赖的对象。抽象工厂模式可以对类的实例化过程进行抽象和封装,并且可以将不同类的实例化分配到不同的具体工厂中,从而使得同一个抽象工厂可以创建出不同的产品对象。

1、写在前面

工厂方法模式通过定义一个工厂接口,将对象实例化的过程抽象出来。但是它存在一个问题,就是一个具体工厂只能创建一类产品,增加新的产品类型时,需要新增具体工厂,违反开闭原则。

那么,如果我们能有一个汽车工厂,既可以生产普通汽车,也可以生产豪华汽车,那岂不美哉?这就是抽象工厂模式要解决的问题。

2、抽象工厂模式的介绍

抽象工厂模式(Abstract Factory Pattern)是一种软件设计模式,它属于创建型模式,其主要目的是创建一组相关或互相依赖的对象。抽象工厂模式可以对类的实例化过程进行抽象和封装,并且可以将不同类的实例化分配到不同的具体工厂中,从而使得同一个抽象工厂可以创建出不同的产品对象。

抽象工厂模式通过抽象层进行解耦,可以独立于产品类的具体实现变化,使得用户可以更加容易地切换不同的产品,而不需要修改已有系统。当需要创建一组相关的对象时,抽象工厂模式尤为合适。它可以确保同一工厂创建的对象是相互兼容、协调的。

3、抽象工厂与其他相关模式

抽象工厂模式与工厂方法模式非常相似,都用于封装对象的创建。区别在于,工厂方法模式中每一个工厂只创建一种产品,而抽象工厂模式可以创建多个相关的产品。抽象工厂也称为工厂的工厂,是工厂方法模式的升级版本。

与建造者模式的区别在于,建造者模式更注重零件的装配顺序,而抽象工厂模式更注重组合,不太关心创建步骤。和原型模式比较,原型模式用于创建单个对象,而抽象工厂用于批量创建产品。

4、抽象工厂模式结构与实现

抽象工厂模式的结构

抽象工厂模式包含以下角色:

  • AbstractFactory:抽象工厂接口,用于创建抽象产品对象。
  • ConcreteFactory:具体工厂实现,实现抽象工厂接口。
  • AbstractProduct:抽象产品接口,定义产品规范。
  • Product:具体产品实现,实现抽象产品接口。
  • Client:使用不同具体工厂和产品的客户。

抽象工厂模式的实现

我们用一个汽车制造的例子来说明抽象工厂的实现。这里的产品分为两大类,豪华品牌和普通品牌,每个品牌都有对应的轿车和越野车产品。

定义抽象工厂和产品接口:

interface AbstractFactory {
  createSedan(): Sedan;
  createSUV(): SUV;
}

interface Sedan {
  showInfo(): void;
}

interface SUV {
  showInfo(): void;  
}

实现具体工厂:

class LuxuryFactory implements AbstractFactory {

  createSedan() {
    return new LuxurySedan();
  }  

  createSUV() {
    return new LuxurySUV();
  }

}

class NormalFactory implements AbstractFactory {

  createSedan() {
    return new NormalSedan();
  }

  createSUV() {
    return new NormalSUV();
  }

}

实现具体产品:

class LuxurySedan implements Sedan {

  showInfo() {
    console.log('展示豪华轿车信息');
  }

}

class LuxurySUV implements SUV {

  showInfo() {
    console.log('展示豪华 SUV 信息');
  }

}

客户端代码:

const factory = new LuxuryFactory();

const sedan = factory.createSedan();
const suv = factory.createSUV();  

sedan.showInfo();
suv.showInfo();

这样客户可以非常方便地切换不同的工厂来获取产品,而不需要关心产品的具体实现。

5、抽象工厂模式的优缺点分析

使用抽象工厂模式带来的优点包括:

  • 封装了具体产品的创建过程,客户端无需知道实现
  • 可以灵活切换不同的产品组合,提高了灵活性
  • 易于增加新产品,满足开闭原则
  • 抽象层实现了解耦,防止源码泄露

但是也存在一些缺点:

  • 过度使用会增加系统的复杂性
  • 增加新的产品时,需要修改抽象工厂和所有具体工厂
  • 产品无法实现继承扩展,都在抽象层定义

所以使用时需要权衡灵活性和复杂性之间的关系,适度使用抽象工厂模式。

6、抽象工厂模式应用于什么场景

抽象工厂模式应用的主要场景包括:

  • 当需要创建的对象是一组相关的产品族时,如电器包含电视、洗衣机等
  • 系统需要多个产品系列,而使用者只需要使用其中某一系列的产品时,如不同品牌的家电
  • 当需要屏蔽用户与产品的具体实现时,使得用户不依赖产品类代码时
  • 当产品类经常变更,而不想影响使用者时
  • 当提供一个产品类库,而只想显示其中部分产品时
  • 需要生成不同平台下的程序时,如Windows、Linux等

在这些情况下,使用抽象工厂模式可以带来很大的灵活性,使得用户可以方便切换不同的产品,而不需要修改已有代码。

抽象工厂模式通过提供一个创建一组相关对象的接口,将客户端与对象的具体实现解耦,使得把兼容的对象组合在一起变得更加容易。当添加新的产品对象时,无需修改已有系统,满足开闭原则。

适用于需要创建一组相关的对象,提供最大化的灵活性和复用性的场景。但也要注意合理使用,过度使用会增加系统的复杂度。

7、总结

抽象工厂模式的主要优点是封装了对象的创建过程,提高了系统的灵活性,可以轻松切换不同的产品配置。使用者无需知道具体实现。另外它也符合开闭原则,容易扩展新产品。

缺点在于过度使用会增加系统的复杂性和抽象性。而且新增产品时需要修改抽象工厂接口,不太容易实现产品的继承扩展。

主要适用于需要创建产品族的场景,需要屏蔽产品具体实现的场景,以及产品配置经常变化的场景。

总体来说,抽象工厂模式在保持系统灵活性和可扩展性方面意义重大。但也需要权衡增加的抽象性带来的复杂度。

责任编辑:姜华 来源: 宇宙一码平川
相关推荐

2023-08-04 07:26:55

工厂类集中化设计模式

2023-08-05 13:31:20

工厂方法模式对象

2023-08-08 20:13:36

设计模式原型模式

2023-08-03 08:01:27

单例模式结构开发

2023-08-15 11:07:37

适配器模式TypeScript

2023-07-30 15:14:19

Koa葱圈模型

2023-08-07 08:01:09

Vuewebpack开发

2023-08-02 08:01:14

柯里化反柯里化

2023-07-27 15:04:10

Node.js核心API

2021-03-06 22:50:58

设计模式抽象

2020-10-19 09:28:00

抽象工厂模式

2021-09-29 13:53:17

抽象工厂模式

2009-01-15 10:55:29

JavaScript设计模式抽象工厂

2020-12-17 09:38:16

设计模式参数

2011-07-28 09:50:58

设计模式

2024-09-14 08:24:44

设计模式抽象工厂模式JDK

2024-03-06 13:19:19

工厂模式Python函数

2022-01-12 13:33:25

工厂模式设计

2020-08-21 07:23:50

工厂模式设计

2011-11-17 16:03:05

Java工厂模式Clojure
点赞
收藏

51CTO技术栈公众号