Java二十三种设计模式-工厂方法模式(2/23)

工厂方法模式:设计模式中的瑞士军刀

引言

在软件开发中,工厂方法模式是一种常用的创建型设计模式,它用于处理对象的创建,将对象的实例化推迟到子类中进行。这种模式不仅简化了对象的创建过程,还提高了代码的可维护性和可扩展性。

基础知识,java设计模式总体来说设计模式分为三大类:

(1)创建型模式,共5种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式。

(2)结构型模式,共7种:适配器模式、装饰器模式、代理模式、外观模式、桥接模式、组合模式、享元模式。

(3)行为型模式,共11种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。

第一部分:工厂方法模式概述

1.1 工厂方法模式定义

工厂方法模式是一种创建型设计模式,用于将对象的创建过程封装在具体的工厂类中,而不是在客户端代码中直接创建。这种模式的目的是将对象创建的逻辑和使用逻辑分离,从而提高代码的可维护性和灵活性。

基本定义

工厂方法模式定义了一个用于创建对象的接口,让子类决定实例化哪一个类。Factory Method让类的实例化推迟到子类进行。

与简单工厂模式的区别

简单工厂模式使用一个中心化的工厂类来创建所有的对象,客户端代码直接与这个工厂类交互以获取所需的对象。这种方式虽然可以简化对象的创建过程,但随着产品种类的增加,工厂类的责任会变得越来越重,难以维护,且违反了开闭原则(对扩展开放,对修改封闭)。

工厂方法模式通过定义一个创建对象的接口,让具体的工厂类实现这个接口,从而为每个产品提供一个具体的工厂,这样新增产品种类时,只需要增加一个具体的工厂类,不需要修改已有代码,更好地遵循了开闭原则。

1.2 工厂方法模式组成

抽象产品(Product)

抽象产品是产品类的共同父类,定义了产品的接口。它是一个抽象角色,不实现具体的产品类。

具体产品(Concrete Product)

具体产品是抽象产品的子类,实现了抽象产品接口中定义的方法。在工厂方法模式中,具体产品由具体工厂类创建。

抽象工厂(Creator)

抽象工厂定义了创建对象的接口。它是一个抽象角色,不实现具体的创建方法。

具体工厂(Concrete Creator)

具体工厂是抽象工厂的子类,实现了抽象工厂中的创建方法。它负责实例化具体产品。

客户端(Client)

客户端代码使用抽象工厂来请求创建对象,客户端不直接与具体产品或具体工厂交互,而是通过抽象工厂与具体工厂进行交互。

通过这种分离,工厂方法模式提供了一种灵活的方式来管理对象的创建,使得新增产品种类时,不需要修改已有的客户端代码,只需要增加相应的具体工厂类即可。这种模式在实际开发中非常实用,尤其是在产品种类较多且经常变化的情况下。在下一部分中,我们将通过Java代码示例来展示工厂方法模式的具体实现。

第二部分:工厂方法模式实现

2.1 Java实现示例

以下是使用Java语言实现工厂方法模式的一个简单示例。我们将创建一个用于生成不同类型汽车的工厂。

// 抽象产品:汽车
interface Car {
    void assemble();
}

// 具体产品:运动型汽车
class SportsCar implements Car {
    public void assemble() {
        System.out.println("Assembling a Sports Car.");
    }
}

// 具体产品:豪华型汽车
class LuxuryCar implements Car {
    public void assemble() {
        System.out.println("Assembling a Luxury Car.");
    }
}

// 抽象工厂:汽车工厂
interface CarFactory {
    Car createCar();
}

// 具体工厂:运动型汽车工厂
class SportsCarFactory implements CarFactory {
    public Car createCar() {
        return new SportsCar();
    }
}

// 具体工厂:豪华型汽车工厂
class LuxuryCarFactory implements CarFactory {
    public Car createCar() {
        return new LuxuryCar();
    }
}

// 客户端代码
public class Client {
    public static void main(String[] args) {
        CarFactory factory = null;

        // 根据需求选择不同类型的汽车工厂
        factory = new SportsCarFactory();
        Car car = factory.createCar();
        car.assemble();

        factory = new LuxuryCarFactory();
        car = factory.createCar();
        car.assemble();
    }
}

2.2 模式中的角色和职责

抽象产品(Product)

  • 职责:定义一个产品接口,规范具体产品类应实现的行为。

具体产品(Concrete Product)

  • 职责:实现抽象产品接口的具体类,是工厂方法模式中被创建的对象。

抽象工厂(Creator)

  • 职责:声明一个创建产品的接口,不涉及具体类实例化。

具体工厂(Concrete Creator)

  • 职责
    • 实现抽象工厂的接口。
    • 具体实现创建产品的方法,生成具体产品对象。

客户端(Client)

  • 职责
    • 根据需求选择相应的具体工厂。
    • 通过具体工厂间接使用具体产品,实现了解耦。

在工厂方法模式中,客户端不直接与具体产品或具体工厂类交互,而是通过抽象工厂接口与具体工厂进行交互。这样做的好处是,当需要添加新的产品时,只需增加相应的具体工厂和具体产品类,而无需修改已有的客户端代码,从而遵循了开闭原则。这种模式提高了系统的灵活性和可扩展性,使得维护和扩展变得更加容易。在下一部分中,我们将讨论工厂方法模式的使用场景。

第三部分:工厂方法模式使用场景

3.1 产品族的扩展

当系统中存在一个产品族,且需要对产品族中的每个具体产品进行扩展时,工厂方法模式提供了一种非常合适的解决方案。

  • 产品族:指一组具有相同接口的产品,这些产品通常在功能上有一定的相似性,但又存在一些差异。
  • 应用场景:例如,在一个图形界面库中,可能存在多种不同类型的按钮(如圆形按钮、矩形按钮、带图标的按钮等)。这些按钮都属于按钮产品族,它们都实现了同一个按钮接口,但又各自有不同的实现。

工厂方法模式的应用:

  • 通过定义一个抽象工厂和工厂方法,可以为每个具体的产品提供一个对应的具体工厂。
  • 当需要扩展产品族时,只需增加相应的具体产品类和对应的具体工厂类,无需修改现有的工厂接口或其他工厂类。

3.2 依赖倒置原则

依赖倒置原则是面向对象设计原则之一,它要求高层模块不应依赖于低层模块,两者都应该依赖于抽象。

  • 实现方式:工厂方法模式通过使用抽象产品和抽象工厂,使得客户端代码依赖于抽象而不是具体实现。
  • 好处:这有助于降低模块间的耦合度,提高系统的灵活性和可维护性。

第四部分:工厂方法模式的优点与缺点

4.1 优点

封装性

工厂方法模式隐藏了对象创建的具体细节,客户端不需要知道具体是如何创建对象的。

扩展性

当需要添加新的产品时,只需增加相应的具体产品和工厂类,无需修改现有代码。

代码复用

通过使用工厂方法模式,可以在一个工厂类中创建多个相似对象,提高代码复用性。

4.2 缺点

增加系统复杂度

每增加一个产品类别,都需要增加一个具体类和产品类,这可能会导致系统中类的数量急剧增加。

增加系统的抽象性

在添加新的产品类时,需要对工厂方法的接口进行扩展,这可能会导致工厂方法的接口经常变化,增加了系统的抽象性。

不容易维护

当工厂方法模式的类数量逐渐增多时,可能会使系统难以理解和维护。

通过深入分析工厂方法模式的使用场景、优点和缺点,我们可以更明智地决定何时以及如何使用工厂方法模式。在实际开发中,我们应该根据具体需求和上下文来选择最合适的设计模式。在下一部分中,我们将比较工厂方法模式与其他设计模式,并提供一些最佳实践和建议。

第五部分:工厂方法模式与其他模式的比较

5.1 与抽象工厂模式的比较

工厂方法模式

  • 定义:用于创建一系列相关或依赖对象的接口,而不需要指定具体类。
  • 使用场景:当一个类不知道它所必须创建的对象的类时。

抽象工厂模式

  • 定义:创建一个工厂接口,用于创建一系列相关的或依赖的抽象产品。
  • 使用场景:当需要创建的产品族是相互关联的,并且希望它们能够一起被使用时。

不同点

  • 产品种类:工厂方法模式创建一个产品,而抽象工厂模式创建产品族。
  • 复杂性:抽象工厂模式通常比工厂方法模式复杂,因为它需要定义更多的接口和类。

5.2 与建造者模式的比较

工厂方法模式

  • 关注点:对象的创建,隐藏创建逻辑。
  • 适用情况:当创建过程只需要一个简单的步骤即可完成时。

建造者模式

  • 关注点:构建一个复杂对象,同时允许用户只通过指定复杂对象的类型和内容就能构建它们。
  • 适用情况:当创建过程需要多步骤或条件逻辑时。

差异

  • 复杂性:建造者模式适用于创建更复杂的对象,而工厂方法模式适用于简单的创建过程。
  • 灵活性:建造者模式提供了更多的灵活性,允许逐步构建对象。

第六部分:最佳实践和建议

6.1 使用工厂方法模式的最佳时机

  • 产品族扩展:当需要创建的产品属于同一类别,并且可能需要扩展时。
  • 依赖关系:当客户端不应该知道具体的产品类时,通过工厂方法模式可以降低客户端与具体产品的耦合。

6.2 避免滥用工厂方法模式

  • 过度使用:避免在不需要对象创建逻辑封装的情况下使用工厂方法模式,这可能导致不必要的复杂性。
  • 难以维护:随着产品种类的增加,管理大量的工厂类可能会变得困难。

6.3 替代方案

原型模式

  • 定义:使用原型实例指定创建对象的种类,并通过复制这些原型创建新的对象。
  • 适用场景:当创建新对象的成本较高时,或者需要通过复制现有的对象来快速创建新对象时。

依赖注入

  • 定义:通过外部注入依赖对象,而不是在类内部创建。
  • 好处:提高了代码的可测试性和灵活性。

服务定位器模式

  • 定义:当需要访问一个服务时,客户端会查询服务定位器以获取服务实例。
  • 适用场景:当系统中有大量的服务需要被访问,并且服务实例的创建和管理较为复杂时。

通过比较工厂方法模式与其他设计模式,我们可以更清晰地了解每种模式的适用场景和特点。在实际开发中,选择最合适的设计模式对于构建一个灵活、可维护的系统至关重要。同时,了解工厂方法模式的最佳实践和替代方案,可以帮助我们更好地应对不同的开发挑战。

结语

工厂方法模式是一种强大且灵活的设计模式,适用于多种不同的软件开发场景。通过本文的深入分析,希望读者能够对工厂方法模式有更全面的理解,并在实际开发中做出合理的设计选择。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/798602.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

WordPress:无法创建新文章?创建新帖子时候页面空白

wordPress中我们新建文章的时候,会遇到页面空白,这个问题是怎么导致呢?我们可以打开F12开发者模式看下报错信息,这是一个警告信息 Warning: Creating default object from empty value in /pub 到数据库 wp_posts中查看生成了很…

SpringBoot新手快速入门系列教程十一:自动生成API文档,Springboot3.x集成SpringDoc

本次项目我们用Maven来做,最近发现gradle其实很多项目的支持比较差,所以项目还是用Maven来新建项目。对比了市面上的几种API生成第三方库,只有springdoc 是能够按照文档就能部署出来的。 官网: OpenAPI 3 Library for spring-bo…

Mac电脑下运行java命令行出现:错误: 找不到或无法加载主类

mac 电脑 问题复现 随手写了一个main方法,想用命令行操作 进入 BlockDemo.java 所在目录: wnwangnandeMBP wn % cd /Users/wn/IdeaProjects/test/JianZhiOffer/src/main/java/com/io/wn wnwangnandeMBP wn % ls -l total 16 -rw-r--r-- 1 wangnan …

Qt文件下载工具

在Qt中实现文件下载功能,通常可以通过多种方式来完成,包括使用 QNetworkAccessManager 和 QNetworkReply 类,或者使用更高级别的 QHttpMultiPart 类。以下是两种常见的实现方法: 方法1:使用 QNetworkAccessManager 和…

芋道框架万字详解(前后端分离)、若依框架、yudao-cloud保姆级攻略

♥️作者:小宋1021 🤵‍♂️个人主页:小宋1021主页 ♥️坚持分析平时学习到的项目以及学习到的软件开发知识,和大家一起努力呀!!! 🎈🎈加油! 加油&#xff01…

tessy 集成测试:小白入门指导手册

目录 1,创建集成测试模块且分析源文件 2,设置测试环境 3,TIE界面设置相关函数 4,SCE界面增加用例 5,编辑数据 6,用例所对应的测试函数序列 7,添加 work task 函数 8,为测试场景添加函数 9,为函数赋值 10,编辑时间序列的数值 11,执行用例 12,其他注意事项…

Linux 下使用Docker安装redis

redis: 是一个高性能的,键值对的,将数据存储到内存中的非关系型数据库(nosql数据库 not only sql) 高性能:数据存在内存中,直接访问内存 键值对:新闻id(键&#xff09…

C语言 | Leetcode C语言题解之第235题二叉搜索树的最近公共祖先

题目&#xff1a; 题解&#xff1a; struct TreeNode* lowestCommonAncestor(struct TreeNode* root, struct TreeNode* p, struct TreeNode* q) {struct TreeNode* ancestor root;while (true) {if (p->val < ancestor->val && q->val < ancestor-&g…

51单片机6(P0P1P2P3结构框架图)

一、GPIO结构框架图与工作原理 1、接下来我们介绍一下这个GPIO结构框图和工作原理&#xff0c;我们使用51单片机的GPIO分为了P0&#xff0c;P1&#xff0c;P2&#xff0c;P3这四组端口&#xff0c;下面我们就分别来介绍这四组端口它的一个内部结构&#xff0c;只有了解了内部的…

windows环境下基于3DSlicer 源代码编译搭建工程开发环境详细操作过程和中间关键错误解决方法说明

说明: 该文档适用于  首次/重新 搭建3D-Slicer工程环境  Clean up(非增量) 编译生成 1. 3D-slicer 软件介绍 (1)3D Slicer为处理MRI\CT等图像数据软件,可以实行基于MRI图像数据的目标分割、标记测量、坐标变换及三维重建等功能,其源于3D slicer 4.13.0-2022-01-19开…

好用的文献翻译软件

自从我开始使用七星文献翻译阅读器&#xff08;https://lestore.lenovo.com/detail/L116395&#xff09;&#xff0c;我的学术世界仿佛被打开了一扇全新的大门。这款软件不仅承载了让知识无国界的宏伟使命&#xff0c;更以其实惠的价格和强大的功能&#xff0c;成为了我学习和研…

【操作系统】进程管理——死锁(个人笔记)

学习日期&#xff1a;2024.7.13 内容摘要&#xff1a;死锁的概念和三大处理策略 目录 死锁 死锁的概念 死锁、饥饿和死循环的区别 死锁产生的必要条件 死锁的处理策略&#xff1a;预防、避免和解除 预防死锁 破坏互斥条件 破坏不剥夺条件 破坏请求和保持条件 破坏循…

【从0到1进阶Redis】主从复制 — 主从机宕机测试

上一篇&#xff1a;【从0到1进阶Redis】主从复制 测试&#xff1a;主机断开连接&#xff0c;从机依旧连接到主机的&#xff0c;但是没有写操作&#xff0c;这个时候&#xff0c;主机如果回来了&#xff0c;从机依旧可以直接获取到主机写的信息。 如果是使用命令行&#xff0c;来…

STM32MP135裸机编程:唯一ID(UID)、设备标识号、设备版本

0 资料准备 1.STM32MP13xx参考手册1 唯一ID&#xff08;UID&#xff09;、设备标识号、设备版本 1.1 寄存器说明 &#xff08;1&#xff09;唯一ID 唯一ID可以用于生成USB序列号或者为其它应用所使用&#xff08;例如程序加密&#xff09;。 &#xff08;2&#xff09;设备…

笔记 1 : 课本前 2 章

现在开始跟着彭老师学习 arm 。把重要的知识点归拢一下&#xff0c;便于复习。早日学有所成&#xff0c;为国为家为己&#xff0c;更幸福些。 &#xff08;1&#xff09;冯诺依曼架构与哈弗架构&#xff0c;与混合架构&#xff1a; 以及&#xff1a; &#xff08;2&#xff0…

php反序列化--2--PHP反序列化漏洞基础知识

一、什么是反序列化&#xff1f; 反序列化是将序列化的字符串还原为PHP的值的过程。 二、如何反序列化 使用unserialize()函数来执行反序列化操作 代码1&#xff1a; $serializedStr O:8:"stdClass":1:{s:4:"data";s:6:"sample";}; $origina…

03-Charles实战

一、抓包分析问题示例 1&#xff09;问题描述 2&#xff09;抓包分析 这是后台响应回来的错误信息&#xff0c;说明问题一是后台的原因&#xff1b;但是后台只响应了一条信息&#xff0c;而前端页面却显示两条错误信息&#xff0c;说明前端页面处理异常的时候逻辑有问题&#…

Android-- 集成谷歌地图

引言 项目需求需要在谷歌地图&#xff1a; 地图展示&#xff0c;设备点聚合&#xff0c;设备站点&#xff0c;绘制点和区域等功能。 我只针对我涉及到的技术做一下总结&#xff0c;希望能帮到开始接触谷歌地图的伙伴们。 集成步骤 1、在项目的modle的build.gradle中添加依赖如…

深层神经网络示例

维度说明&#xff1a; A[L]、Z[L]&#xff1a;&#xff08;本层神经元个数、样本数&#xff09; W[L]&#xff1a;&#xff08;本层神经元个数、上层神经元个数&#xff09; b[L]&#xff1a;&#xff08;本层神经元个数、1&#xff09; dZ[L]&#xff1a;dA[L] * g’A&#xf…

leetcode 513. 找树左下角的值

给定一个二叉树的 根节点 root&#xff0c;请找出该二叉树的 最底层 最左边 节点的值。 假设二叉树中至少有一个节点。 示例 1: 输入: root [2,1,3] 输出: 1示例 2: 输入: [1,2,3,4,null,5,6,null,null,7] 输出: 7提示: 二叉树的节点个数的范围是 [1,104]-231 < Node.val &…