面试笔记——工厂模式(简单工厂、工厂方法模式、抽象工厂模式)

2024-05-07 18:44

本文主要是介绍面试笔记——工厂模式(简单工厂、工厂方法模式、抽象工厂模式),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

场景需求:设计一个咖啡店点餐系统。 设计一个咖啡类(Coffee),并定义其两个子类(美式咖啡【AmericanCoffee】和拿铁咖啡【LatteCoffee】);再设计一个咖啡店类(CoffeeStore),咖啡店具有点咖啡的功能。

在不使用工厂模式情况下,类设计如下:
在这里插入图片描述
以上类图中的符号表示:

  • +:表示public
  • -:表示private
  • #:表示protected
  • 泛化关系(继承) 用带空心三角箭头的实线来表示
  • 实现关系 用带空心的三角箭头的虚线来表示
  • 依赖关系 使用带箭头的虚线来表示
  • 类图的分成三部分:第一部分——类的名称;第二部分——类的属性;第三部分——类的方法,方法的“:”后面表示方法的返回值类型

简单工厂模式

简单工厂包含如下角色:

  • 抽象产品 :定义了产品的规范,描述了产品的主要特性和功能。
  • 具体产品 :实现或者继承抽象产品的子类,每种具体产品都有自己的特点。
  • 具体工厂 :提供了创建产品的方法,调用者通过该方法来获取产品。

简单工厂模式下的类实现:
在这里插入图片描述
CoffeeStore不再关注创建咖啡的类型,创建咖啡的具体实现交给SimpleCoffeeFactory。

简单工厂模式的主要优点

  1. 封装性:客户端代码不需要知道创建对象的具体类,只需要知道相应的参数即可,这降低了客户端代码与具体实现之间的耦合度。

  2. 集中管理:所有的对象创建逻辑都集中在工厂类中,便于管理和维护。

  3. 易于扩展:当需要添加新的产品类型时,只需要在工厂类中添加新的逻辑,而不需要修改客户端代码。

简单工厂模式的缺点

  1. 违反开闭原则:当添加新的产品类型时,需要修改工厂类,这违反了开闭原则(软件工程原则之一,即软件实体应当是可扩展,而不可修改的)。

  2. 工厂类职责过重:随着产品类型的增多,工厂类的职责可能会变得过于繁重,这可能导致代码的可读性和可维护性下降。

  3. 类型安全问题:由于工厂方法通常使用字符串参数来确定要创建的对象类型,这可能导致类型安全问题,比如传入无效的参数。

工厂方法模式

工厂方法模式的主要角色:

  • 抽象工厂(Abstract Factory):提供了创建产品的接口,调用者通过它访问具体工厂的工厂方法来创建产品。
  • 具体工厂(Concrete Factory):主要是实现抽象工厂中的抽象方法,完成具体产品的创建。
  • 抽象产品(Product):定义了产品的规范,描述了产品的主要特性和功能。
  • 具体产品(Concrete Product):实现了抽象产品角色所定义的接口,由具体工厂来创建,它同具体工厂之间一一对应。

工厂方法模式下类的具体设计:
在这里插入图片描述
Coffee类是抽象产品角色;AmericanCoffee和LatteCoffee类是具体产品角色;CoffeeFactory是抽象工厂角色,定义了创建产品的规范;AmericanCoffeeFactory和LatteCoffeeFactory是具体工厂角色。

此设计模式下的调用关系:
在这里插入图片描述
为了便于理解,CoffeeStore的代码如下:

public class CoffeeStore {public static void main(String[] args) {//可以根据不同的工厂,创建不同的产品CoffeeStore coffeeStore = new CoffeeStore(new LatteCoffeeFactory());Coffee latte = coffeeStore.orderCoffee();System.out.println(latte.getName());}private CoffeeFactory coffeeFactory;public CoffeeStore(CoffeeFactory coffeeFactory){this.coffeeFactory = coffeeFactory;}public Coffee orderCoffee(){Coffee coffee = coffeeFactory.createCoffee();//添加配料coffee.addMilk();coffee.addSuqar();return coffee;}
}

如上面的代码所示,用户只需要根据需求,创建一个具体工厂对象就可以了。

工厂方法模式的优点

  1. 解耦:用户只需要知道具体工厂的名称就可以得到所要的产品,无需知道产品的具体创建过程。
  2. 扩展性:当系统需要引入新的产品时,只需要添加新的具体产品类和对应的具体工厂类,无需修改原有的工厂类,这符合开闭原则
  3. 灵活性:工厂方法模式提供了一种灵活的方式来创建对象,解除了客户端与具体产品的依赖关系,增加了代码的可扩展性和可维护性。

工厂方法模式的缺点

  1. 增加系统复杂度:每增加一个产品,就需要增加一个具体产品类和对应的具体工厂类,这增加了系统的复杂度、开发量和维护成本。。
  2. 运行效率低:每次创建具体产品类的实例时,都需要通过具体工厂类来创建,这可能会影响系统的运行效率。

抽象工厂模式

抽象工厂模式是工厂方法模式的升级版本,工厂方法模式只生产一个等级的产品,而抽象工厂模式可生产多个等级的产品。
一个超级工厂创建其他工厂。该超级工厂又称为其他工厂的工厂。

关于产品和等级的理解:
在这里插入图片描述
产品族:一个品牌下面的所有产品;例如华为下面的电脑、手机称为华为的产品族;
产品等级:多个品牌下面的同种产品;例如华为和小米都有手机电脑为一个产品等级;

工厂方法模式只考虑生产同等级的产品,抽象工厂可以处理多等级产品的生产。

使用场景:现咖啡店业务发生改变,不仅要生产咖啡还要生产甜点

  • 同一个产品等级(产品分类)
    • 咖啡:拿铁咖啡、美式咖啡
    • 甜点:提拉米苏、抹茶慕斯
  • 同一个风味,就是同一个产品族(相当于同一个品牌)
    • 美式风味:美式咖啡、抹茶慕斯
    • 意大利风味:拿铁咖啡、提拉米苏

类的具体实现:
在这里插入图片描述
产品工厂就是超级工厂角色。
此时的调用关系为:
在这里插入图片描述
抽象工厂模式的优点

  1. 客户端无需知道它所需要的对象的类,只需要知道相应的工厂即可,这降低了客户端与具体实现之间的耦合度。
  2. 可以在类的内部对产品族中相关联的多等级产品共同管理,而不必专门引入多个新的类来进行管理。
  3. 当需要引入新的产品族时,只需要添加新的工厂类和对应的具体产品类,而无需修改已有的代码,这符合开闭原则。
  4. 当一个产品族中的多个对象被设计成一起工作时,它能保证客户端始终只使用同一个产品族中的对象。

抽象工厂模式的缺点

  1. 产品族扩展困难,需要修改工厂类的代码来支持新的产品族。

综上:

  • 简单工厂(并不算是设计模式)
    • 所有的产品都共有一个工厂,如果新增产品,则需要修改代码,违反开闭原则
    • 是一种编程习惯,可以借鉴这种编程思路
  • 工厂方法模式
    • 给每个产品都提供了一个工厂,让工厂专门负责对应的产品的生产,遵循开闭原则
    • 项目中用的最多
  • 抽象工厂方法模式
    • 如果有多个纬度的产品需要配合生产时,优先建议采用抽象工厂(工厂的工厂)
    • 一般的企业开发中的较少

这篇关于面试笔记——工厂模式(简单工厂、工厂方法模式、抽象工厂模式)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/968080

相关文章

PHP轻松处理千万行数据的方法详解

《PHP轻松处理千万行数据的方法详解》说到处理大数据集,PHP通常不是第一个想到的语言,但如果你曾经需要处理数百万行数据而不让服务器崩溃或内存耗尽,你就会知道PHP用对了工具有多强大,下面小编就... 目录问题的本质php 中的数据流处理:为什么必不可少生成器:内存高效的迭代方式流量控制:避免系统过载一次性

python获取指定名字的程序的文件路径的两种方法

《python获取指定名字的程序的文件路径的两种方法》本文主要介绍了python获取指定名字的程序的文件路径的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 最近在做项目,需要用到给定一个程序名字就可以自动获取到这个程序在Windows系统下的绝对路径,以下

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

504 Gateway Timeout网关超时的根源及完美解决方法

《504GatewayTimeout网关超时的根源及完美解决方法》在日常开发和运维过程中,504GatewayTimeout错误是常见的网络问题之一,尤其是在使用反向代理(如Nginx)或... 目录引言为什么会出现 504 错误?1. 探索 504 Gateway Timeout 错误的根源 1.1 后端

Python 基于http.server模块实现简单http服务的代码举例

《Python基于http.server模块实现简单http服务的代码举例》Pythonhttp.server模块通过继承BaseHTTPRequestHandler处理HTTP请求,使用Threa... 目录测试环境代码实现相关介绍模块简介类及相关函数简介参考链接测试环境win11专业版python

MySQL 表空却 ibd 文件过大的问题及解决方法

《MySQL表空却ibd文件过大的问题及解决方法》本文给大家介绍MySQL表空却ibd文件过大的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录一、问题背景:表空却 “吃满” 磁盘的怪事二、问题复现:一步步编程还原异常场景1. 准备测试源表与数据

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

SpringBoot通过main方法启动web项目实践

《SpringBoot通过main方法启动web项目实践》SpringBoot通过SpringApplication.run()启动Web项目,自动推断应用类型,加载初始化器与监听器,配置Spring... 目录1. 启动入口:SpringApplication.run()2. SpringApplicat