工厂方法模式与抽象工厂模式的深度对比

在这里插入图片描述​🌈 个人主页:danci_
🔥 系列专栏:《设计模式》
💪🏻 制定明确可量化的目标,坚持默默的做事。
🚀 转载自:设计模式深度解析:工厂方法模式与抽象工厂模式的深度对比    已被《松山湖开发者村综合服务平台》收录


设计模式深度解析:工厂方法模式与抽象工厂模式的深度对比

探索设计模式的魅力:工厂方法模式文章浏览阅读17k次,点赞105次,收藏72次。工厂方法模式是一种创建型设计模式,它提供了一种创建对象的接口,但将具体实例化对象的工作推迟到子类中完成。这样做的目的是创建对象时不用依赖于具体的类,而是依赖于抽象,这提高了系统的灵活性和可扩展性。优点:降低耦合度、增加了系统的可扩展性 和 提高代码的可维护性;缺点:增加了代码的复杂性 和 需要更多的设计考虑。https://blog.csdn.net/danci_/article/details/135611783

探索设计模式的魅力:抽象工厂模式的艺术文章浏览阅读14k次,点赞76次,收藏76次。抽象工厂模式(Abstract Factory Pattern)是一种创建型设计模式,用于在不指定具体类的情况下创建一系列相关或相互依赖的对象。它提供了一个接口,用于创建一系列“家族”或相关依赖对象,而无需指定它们的具体类。探索设计模式的魅力:简单工厂模式-CSDN博客实现简单工厂的难点就在于 “如何选择” 实现,前面便子中传递参数的方法, 那都是静态的参数,还可以实现成为动态的参数。https://blog.csdn.net/danci_/article/details/135638488

文章目录

  • 一、定义🌐
    • 模式对比</font></code>
  • 二、结构图🔍
    • 参与者👇
    • 适用场景
  • <code>三、易混场景💔<code>
    • 场景
    • 工厂方法模式
    • 抽象工厂模式
    • 易混淆之处</code>
  • 五、总结 💖
    • 工厂方法模式
    • 工厂模式最佳实践和使用场景
    • 抽象工厂模式关键点
    • 抽象工厂模式最佳实践和使用场景
    • 根据项目需求选用正确的模式的建议</code>
    • 应用考量</code>

一、定义🌐

工厂方法模式

定义一个用于创建对象的接口,将具体实例化对象的工作推迟到子类中完成。

 作用

当系统需要引入新的产品类型时,只需要增加相应的工厂子类,而不需要修改原有的系统代码,从而实现了“开闭原则”。

 如何封装对象的创建过程
    在工厂方法模式中,对象的创建过程被封装在工厂类的工厂方法中。具体来说,这个过程包括以下几个步骤:👇
 1. 定义抽象产品接口:
    首先,需要定义一个抽象产品接口,该接口描述了所有具体产品类应该具有的方法。

 2. 定义抽象工厂类:
    然后,定义一个抽象工厂类,该类声明了一个工厂方法,用于创建抽象产品接口的对象。这个工厂方法通常被声明为抽象方法,以便子类可以实现它。

 3. 实现具体产品类:
    接下来,根据抽象产品接口定义具体的产品类。这些类实现了抽象产品接口中声明的方法,并提供了具体的产品实现。

 4. 实现具体工厂类:
    然后,创建具体工厂类,该类继承自抽象工厂类,并实现工厂方法。在工厂方法中,具体工厂类根据需要创建并返回相应的具体产品对象。

    通过这种方式,工厂方法模式封装了对象的创建过程。客户端代码只需要知道抽象产品接口和抽象工厂类,而无需关心具体产品类的实现和对象的创建过程。这有助于降低代码的耦合度,提高系统的可扩展性和可维护性。同时,由于具体产品的创建逻辑是由具体工厂类来实现的,因此也增加了系统的灵活性,使得系统可以更容易地引入新的产品类型。

抽象工厂模式

提供了一个创建一系列相关或相互依赖对象的接口,用于创建相关或依赖对象的家族,而不需要明确指定具体类。

 作用

通过抽象化对象的创建过程,实现高内聚低耦合的设计原则,增强系统的可扩展性和可维护性。。

 如何封装对象的创建过程
    在抽象工厂模式中,每个工厂都能够创建一系列相互关联或依赖的对象,而客户端代码则通过抽象接口与这些工厂进行交互,从而无需了解具体对象的创建逻辑。抽象工厂模式通过以下几个关键步骤来封装对象的创建过程:👇
 1. 定义抽象产品接口:
    首先,为每种类型的产品定义一个抽象接口。这些接口将规定所有具体产品类必须实现的方法。

 2. 定义抽象工厂接口:
    接下来,定义一个抽象工厂接口,该接口将声明一组创建抽象产品的方法。这些方法通常对应于步骤1中定义的抽象产品接口。

 3. 实现具体产品类:
    根据步骤1中定义的抽象产品接口,创建具体的产品类。这些类将实现抽象产品接口中声明的方法,并提供具体的产品实现。

 4. 实现具体工厂类:
    创建具体工厂类,该类将实现抽象工厂接口。在具体工厂类中,实现抽象工厂接口中声明的方法,以便它们能够创建和返回相应的具体产品对象。

 5. 客户端代码使用:
    在客户端代码中,通过创建具体工厂类的对象并使用它来创建产品对象。客户端代码仅依赖于抽象产品接口和抽象工厂接口,因此与具体产品类和具体工厂类的实现细节解耦。

模式对比

    为便于对比理解,总结如下图所示:
在这里插入图片描述
    工厂模式定义了一个用于创建对象的接口,让子类决定实例化哪一个类。它将一个类的实例化延迟到子类进行,这样可以将对象的创建与使用分离,使代码更加灵活和可维护。在工厂模式中,通常会有一个具体的工厂类负责创建某一类产品,客户端直接调用工厂类的方法来获取所需的对象。

    抽象工厂模式则提供了一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。这个模式允许客户端在不指定具体产品类的情况下创建多个产品族中的产品对象。抽象工厂模式是所有形态的工厂模式中最为抽象和最具有一般性的一种形态。在抽象工厂模式中,工厂类不再仅仅负责创建单一产品,而是负责创建一组具有相同主题或相互依赖的产品对象。这意味着抽象工厂模式通常涉及多个抽象产品和多个具体产品,需要定义更多的接口和类。

    进一步来说,工厂方法模式与抽象工厂模式的区别主要体现在创建对象的范围和复杂度上。工厂模式主要关注单一产品等级的创建,而抽象工厂模式则强调多个产品族中相关对象的创建。从结构上看,工厂模式相对简单,而抽象工厂模式则更为复杂,涉及更多的接口和类定义。

二、结构图🔍

在这里插入图片描述

参与者👇

 1. 工厂方法模式:
    产品(Product): 定义工厂方法所创建的对象的接口。所有被创建的对象都是某个具体产品的实例。

    具体产品(Concrete Product): 实现产品接口的具体类。工厂方法模式所创建的每个对象都是某个具体产品类的实例。

    创建者(Creator): 声明工厂方法,该方法返回一个产品类型的对象。创建者可能提供工厂方法的默认实现,它返回一个默认的具体产品对象。

    具体创建者(Concrete Creator): 重写工厂方法以返回一个具体产品实例。每个具体创建者通常与一个特定的具体产品一一对应。

    工厂方法模式通过抽象工厂类和具体工厂类的协作,实现了对象的延迟实例化,降低了系统的耦合度,提高了系统的可维护性和扩展性。同时,它还利用了多态性,使得代码更加灵活和可重用。

 2. 抽象工厂模式:
    抽象工厂(Abstract Factory): 它声明了一组用于创建一系列产品的方法,每个方法都对应一个产品等级。抽象工厂可以看作是一个工厂的工厂,或者是一系列工厂的抽象。在抽象工厂中,每个方法对应一种产品,实现了对产品的集中管理。客户端使用抽象工厂来获取产品时,无需关心具体实现,只需通过调用相应的方法即可获取到产品。

    具体工厂(Concrete Factory):它实现了抽象工厂中声明的创建产品的方法,以生产具有共同主题的具体产品。每个具体工厂负责生成一个产品族中的所有产品,这些产品之间具有一定的关联性或依赖性。当客户端需要某个产品族中的产品时,只需使用相应的具体工厂即可。

    抽象产品(Abstract Product):它是定义产品的接口,描述了所有产品所共有的特性或方法。每个抽象产品对应一个产品等级结构,即同一类产品的不同实现方式。

    具体产品(Concrete Product): 它实现了抽象产品中定义的接口,提供了具体的产品实现。每个具体产品都属于一个产品族,与该产品族中的其他产品具有一定的关联性或依赖性。

    抽象工厂模式适用于需要创建一系列相互关联或相互依赖的产品的场景。它通过将具体工厂的创建过程抽象化,使得客户端可以根据需要灵活地选择不同的产品族进行创建,从而实现了代码的解耦和模块化。然而,抽象工厂模式的缺点是当需要增加新的产品族时,需要修改抽象工厂的接口以及所有具体工厂的实现,这可能会带来较大的改动和维护成本。因此,在使用抽象工厂模式时需要权衡其优点和缺点,根据具体的业务需求和系统规模进行选择。

适用场景

在这里插入图片描述

 

三、易混场景💔

在这里插入图片描述

场景

假设我们正在开发一个游戏,游戏中有多种类型的角色,每种角色都有不同的武器和技能。我们需要设计一个系统来创建和管理这些角色、武器和技能。
     在这个场景中,我们可以考虑使用工厂方法模式或抽象工厂模式来创建角色、武器和技能对象。然而,这两种模式的选择和使用方式可能会引起混淆。

  

工厂方法模式

如果我们选择使用工厂方法模式,我们可以定义一个抽象的角色工厂类,它声明了一个创建角色的工厂方法。然后,我们可以为每种具体的角色类型创建一个具体的角色工厂类,这些工厂类实现了抽象角色工厂类中声明的工厂方法,用于创建具体类型的角色对象。同样地,我们也可以为武器和技能定义类似的工厂类和方法。

    在这种情况下,如果我们需要创建一个具有特定武器和技能的角色对象,我们可能需要分别调用角色工厂、武器工厂和技能工厂的方法。这可能会导致代码中的耦合度增加,因为客户端代码需要知道如何组合使用这些工厂方法来创建完整的角色对象。

抽象工厂模式

如果我们选择使用抽象工厂模式,我们可以定义一个抽象的工厂接口,该接口声明了一组创建角色、武器和技能对象的工厂方法。然后,我们可以为每种具体的角色类型创建一个具体的工厂类,这些工厂类实现了抽象工厂接口中声明的所有工厂方法,用于创建特定类型的角色、武器和技能对象。
     在这种情况下,客户端代码只需要知道如何使用抽象工厂接口即可创建完整的角色对象,而无需关心具体工厂类的实现细节。这有助于降低代码中的耦合度,提高系统的灵活性和可扩展性。然而,当需要添加新的角色类型或新的产品族时,可能需要修改抽象工厂接口和现有的具体工厂类,这可能会带来一些维护成本。

  

易混淆之处

    在这个场景中,工厂方法模式和抽象工厂模式的易混淆之处在于它们都涉及创建多种类型的对象。然而,工厂方法模式侧重于通过继承来实现对象的创建逻辑的封装和扩展,而抽象工厂模式则侧重于通过组合来实现一系列相互关联或相互依赖的产品的创建逻辑的封装和扩展。因此,在选择使用这两种模式时,需要根据具体的系统需求和设计目标来进行权衡和决策。

    注:在实际应用中,这两种模式并不是互斥的,而是可以相互结合使用的。例如,我们可以在抽象工厂模式中使用工厂方法来创建具体的产品对象,从而实现更灵活和可扩展的系统设计。
 

五、总结 💖

在这里插入图片描述

工厂方法模式

 1. 封装性:
    🌈 工厂模式通过专门的工厂类来创建其他类的实例,隐藏了对象创建的具体逻辑,客户端只需要知道产品的抽象接口和工厂类提供的创建方法。

 2. 解耦:
    🌈 工厂模式减少了客户端与具体产品类之间的依赖,客户端通过工厂接口与工厂类交互,无需了解具体实现细节。

 3. 单一职责:
    🌈 每个工厂类通常只负责创建一种或一类产品,符合单一职责原则。

 4. 扩展性:
    🌈 当需要添加新产品时,只需增加相应的具体产品类和对应的工厂类,而无需修改客户端代码。

工厂模式最佳实践和使用场景

 1. 🌈 当需要创建的对象具有复杂的初始化逻辑或依赖于外部资源时,使用工厂模式可以封装这些复杂性。

 2. 🌈 当系统中存在多个类似的产品,且这些产品的创建逻辑可能会变化时,使用工厂模式可以提高系统的灵活性和可维护性。

 3.  🌈 当希望将对象的创建与使用分离,以便在不修改客户端代码的情况下更换产品实现时,使用工厂模式可以实现这一目标。
 

抽象工厂模式关键点

 1. 产品族:
    🚀 抽象工厂模式强调一系列相互关联或相互依赖的产品对象(产品族)的创建,而不仅仅是单个产品的创建。

 2. 一致性:
    🚀 客户端通过抽象工厂接口获取一系列产品对象,确保这些对象在逻辑上是一致的、相互兼容的。

 3. 封装性:
    🚀 抽象工厂模式封装了具体产品族的创建逻辑,客户端只需要与抽象工厂接口交互。

 4. 扩展性:
    🚀 当需要添加新的产品族时,只需增加相应的具体工厂类,而无需修改现有代码(遵循开闭原则)。
 

抽象工厂模式最佳实践和使用场景

 1. 🚀 当系统需要处理多个产品族,并且每个产品族包含多个相互关联的产品时,使用抽象工厂模式可以方便地创建和管理这些产品族

 2. 🚀 当希望确保客户端使用的产品对象来自同一个产品族,以保持逻辑上的一致性时,使用抽象工厂模式可以实现这一目标。

 3. 🚀 当产品的创建逻辑可能会因为不同的平台、配置或环境而有所不同时,使用抽象工厂模式可以方便地切换不同的产品族实现。
 

根据项目需求选用正确的模式的建议

 1. 分析需求:
    ✨ 首先明确项目中需要创建的对象类型以及它们之间的关系。如果只需要创建单一类型的对象,且对象的创建逻辑相对简单,那么工厂模式可能是更好的选择。如果需要创建多个相互关联的对象(产品族),则考虑使用抽象工厂模式。

 2.  考虑扩展性:
    ✨ 评估未来可能的产品变化。如果预计会有新的产品类型或产品族加入,那么选择更具扩展性的模式(如抽象工厂模式)可能更为合适。

 3. 遵循设计原则:
    ✨ 尽量遵循面向对象设计原则,如单一职责原则、开闭原则等。这些原则可以帮助你做出更合理的模式选择。

 4. 代码简洁性:
    ✨ 在满足功能需求的前提下,尽量选择使代码更简洁、易于理解和维护的模式。过度设计可能会增加系统的复杂性和维护成本。

应用考量

 1. 系统的复杂度:
    🌟 如果系统相对简单,产品种类较少且不太可能发生变化,那么简单工厂模式可能是更好的选择。如果系统复杂,存在多个产品族且产品族之间可能存在较大差异,那么抽象工厂模式可能更合适。

 2. 可扩展性需求:
    🌟 如果预计未来需要频繁添加新产品或新产品族,那么抽象工厂模式可能更具扩展性。因为抽象工厂模式可以通过添加新的具体工厂类来引入新产品族,而无需修改现有代码。而简单工厂模式可能需要修改工厂类以适应新产品的创建。

 3. 设计原则:
    🌟 尽量遵循面向对象设计原则,如单一职责原则、开闭原则等。这些原则可以帮助你做出更合理的模式选择。例如,如果希望遵循开闭原则,那么抽象工厂模式可能更合适,因为它允许在不修改现有代码的情况下扩展系统。

 4. 代码简洁性:
    🌟 在满足功能需求的前提下,尽量选择使代码更简洁、易于理解和维护的模式。过度设计可能会增加系统的复杂性和维护成本。因此,在选择设计模式时,需要权衡其带来的好处和可能增加的复杂性。

    ❤️ 选择简单工厂模式还是抽象工厂模式应根据具体项目的需求、复杂度和可扩展性要求来决定。在实际应用中,可以先从简单工厂模式开始,随着项目的发展和需求的变化,再考虑是否升级到抽象工厂模式或其他更复杂的模式。

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

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

相关文章

张颂文|永远保持好奇心的人,是永远进步的人。

哈喽,你好啊,我是雷工! 今天看到了张颂文的一段演讲,提到了他因为好奇心而被改变的人生。 如果想把单一和枯燥的工作做的更好,张颂文的办法是像一个孩子一样保持好奇心,不停地提出一些有趣的问题。 在5年的导游经历中,对每次游览的地点都像初次游览般保持好奇心,正因为…

【C语言】【Leetcode】2437. 有效时间的数目

文章目录 题目思路一、枚举思路二、回溯 题目 链接: link 思路一、枚举 这题的可以简单的看成 h1 h2 : m1 m2 的情况&#xff0c;其中 h1 和 h2 有关&#xff0c; m1 和 m2 有关&#xff0c;数目不多可以直接暴力枚举解决 int countTime(char * time) {int countHour 0;i…

【JavaWeb】Day24.Web入门——SPringBootWeb入门

什么是SPring&#xff1f; 我们可以打开Spring的官网(Spring | Home)&#xff0c;去看一下Spring的简介&#xff1a;Spring makes Java simple。Spring的官方提供很多开源的项目&#xff0c;我们可以点击上面的projects&#xff0c;看到spring家族旗下的项目&#xff0c;按照流…

PLC的大脑和心脏——CPU及西门子S7-1200CPU分类、CPU型号及端子接线图示例

CPU不断地采集输入信号&#xff0c;执行用户程序&#xff0c;刷新系统的输出。 根据供电方式和输入/输出方式的不同&#xff0c;西门子S7-1200 CPU分为3类&#xff0c;如下图1。 图1 CPU的分类 第1对字母&#xff0c;表示CPU的供电方式&#xff0c;AC&#xff08;Alternating…

leetcode 不同路径

62. 不同路径 问题描述 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;。 问总共有多少条不同的…

python_2

文章目录 题目一运行结果 题目二运行结果 题目一 代码如下&#xff1a; def merge():ls_0 input("输入一个列表(空格隔开)&#xff1a;").split()ls_1 []for i in ls_0:ls_1.append(i)ls_1.sort()if ls_0 ls_1:print("这是一个有序列表")else:print(&qu…

国内首个BEV感知全栈系列学习教程:课程总结

目录 前言零. 简述一、BEV感知算法介绍二、BEV感知算法基础模块讲解三、LiDAR和Camera融合的BEV感知算法四、基于环视Camera的BEV感知算法五、BEV感知算法实战总结 前言 自动驾驶之心推出的《国内首个BVE感知全栈系列学习教程》&#xff0c;链接。记录下个人学习笔记&#xff0…

系统慢查询的思考

系统慢查询的思考 在一个系统中发现慢查询的功能或很卡的现象。你是怎么思考的&#xff1f;从哪几个方面去思考&#xff1f;会用什么工具&#xff1f; 一个系统使用了几年后都可能会出现这样的问题。原因可能有以下几点。 数据量的增加。系统中平时的使用中数据量是有一个累…

cdr弧形线条怎么画 cdr弧形线条怎么复制 CoreIDRAW官版 CoreIDRAW2024 平面设计软件

弧形线条可以增加设计的美感和独特性&#xff0c;使其看起来更加优雅和精致&#xff0c;并且弧形线条可以使设计更加流畅&#xff0c;减少直角和生硬的转折&#xff0c;使其看起来更加自然。那在cdr软件中怎么绘制弧形线条呢&#xff1f;下面由我带大家一起来了解cdr弧形线条怎…

c++编程(1)——重载函数、引用

欢迎来到博主的专栏——c编程 博主ID&#xff1a; 代码小豪 文章目录 前言重载函数函数重载的规则函数重载的原理引用引用变量的权限问题 前言 c语言对于编写大型项目有所缺陷&#xff0c;比如最常出现的标识符不能重复的问题&#xff08;软件的代码量通常是数以万计的&#…

机器语言编写helloworld

kvmtool下载编译 git clone https://github.com/kvmtool/kvmtool.git 下载后进入到目录执行make即可。 补码 计算机怎么表示负数&#xff1f;以四位有符号数为例&#xff0c;使用高位作为符号位&#xff0c;最高位为0表示正数&#xff0c;为1表示负数&#xff0c;其余三位用…

基于SSM远程同步课堂系统

基于SSM远程同步课堂系统的设计与实现 摘要 在这样一个网络数据大爆炸的时代&#xff0c;人们获取知识、获取信息的通道非常的多元化&#xff0c;通过网络来实现数据信息的获取成为了现在非常常见的一种方式&#xff0c;而通过网络进行教学&#xff0c;在网络上进行远程的课堂…

【软考】数据流图的设计原则

目录 1. 数据守恒原则2. 守恒加工原则3. 外部实体与外部实体之间不存在数据流4. 外部实体与外部存储之间不存在数据流5. 数据存储与数据存储之间不存在数据流6. 父图与子图的平衡原则7. 数据流与加工有关&#xff0c;且必须经过加工8.例题8.1 例题1 1. 数据守恒原则 1.输入与输…

嵌入式linux学习之opencv交叉编译

1.下载opencv源码 OpenCV官方源码下载链接为https://opencv.org/releases/&#xff0c;选择3.4.16版本下载。放在ubuntu系统~/opencv文件夹中&#xff0c;解压缩&#xff0c;opencv文件夹中新建build和install文件夹用于存放编译文件和安装文件&#xff1a; 2. 安装编译工具…

ES的RestClient相关操作

ES的RestClient相关操作 Elasticsearch使用Java操作。 本文仅介绍CURD索引库和文档&#xff01;&#xff01;&#xff01; Elasticsearch基础&#xff1a;https://blog.csdn.net/weixin_46533577/article/details/137207222 Elasticsearch Clients官网&#xff1a;https://ww…

MD5 计算 (下一代加密辅助类, Win32, C++)

CCNGHelper.h #pragma once #include <string> #include <tchar.h> #include <windows.h> #include <bcrypt.h>#ifdef _UNICODE using _tstring std::wstring; #else using _tstring std::string; #endif// 下一代加密辅助类 // 客户端: Windows Vi…

Vue2(十二):Vuex环境搭建、Vuex工作原理、Vuex开发者工具、几个配置项、多组件共享数据、Vuex模块化

一、Vuex 1.概念 专门在Vue中实现集中式状态&#xff08;数据&#xff09;管理的一个Vue插件&#xff08;use引入&#xff09;&#xff0c;对vue应用中多个组件的共享状态进行集中式的管理&#xff08;读&#xff0f;写&#xff09;&#xff0c;也是一种组件间通信的方式&…

阿里云优惠券领取方法大公开,省钱不再是难事

阿里云作为国内领先的云计算服务提供商&#xff0c;为广大用户提供了丰富的云产品和解决方案。为了吸引用户上云&#xff0c;阿里云经常推出各种优惠活动&#xff0c;其中最受用户欢迎的就是阿里云优惠券。那么&#xff0c;阿里云优惠券究竟是什么呢&#xff1f;我们又该如何领…

代码随想录第25天|216.组合总和III 17.电话号码的字母组合

216.组合总和III 216. 组合总和 III - 力扣&#xff08;LeetCode&#xff09; 代码随想录 (programmercarl.com) 和组合问题有啥区别&#xff1f;回溯算法如何剪枝&#xff1f;| LeetCode&#xff1a;216.组合总和III_哔哩哔哩_bilibili 找出所有相加之和为 n 的 k 个数的组…

基于自动编码器的预训练模型方法模型预训练方法RetroMAE和RetroMAE-2

文章目录 RetroMAERetroMAE详情编码解码增强解码 RetroMAE-2RetroMAE-2详情编码[CLS]解码OT解码和训练目标向量表征 总结参考资料 RetroMAE RetroMAE 出自论文《RetroMAE: Pre-Training Retrieval-oriented Language Models Via Masked Auto-Encoder》&#xff0c;是一种针对于…