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

在这里插入图片描述​🌈 个人主页: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. 代码简洁性:
    🌟 在满足功能需求的前提下,尽量选择使代码更简洁、易于理解和维护的模式。过度设计可能会增加系统的复杂性和维护成本。因此,在选择设计模式时,需要权衡其带来的好处和可能增加的复杂性。

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

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

intellij idea 使用git撤销(取消)commit

git撤销(取消) 未 push的 commit Git&#xff0c;选择分支后&#xff0c;右键 Undo Commit &#xff0c;会把这个 commit 撤销。 git撤销(取消) 已经 push 的 commit 备份分支内容&#xff1a; 选中分支&#xff0c; 新建 分支&#xff0c;避免后续因为操作不当&#xff0c;导…

Spring重点知识(个人整理笔记)

目录 1. 为什么要使用 spring&#xff1f; 2. 解释一下什么是 Aop&#xff1f; 3. AOP有哪些实现方式&#xff1f; 4. Spring AOP的实现原理 5. JDK动态代理和CGLIB动态代理的区别&#xff1f; 6. 解释一下什么是 ioc&#xff1f; 7. spring 有哪些主要模块&#xff1f;…

layui框架实战案例(26):layui-carousel轮播组件添加多个Echarts图标的效果

在Layui中&#xff0c;使用layui-carousel轮播组件嵌套Echarts图表来实现多个图表的展示。 css层叠样式表 调整轮播图背景色为白色&#xff1b;调整当个Echarts图表显示loading…状态&#xff1b;同一个DIV轮播项目添加多个Echarts的 .layui-carousel {background-color: #f…

前端路径问题总结

1.相对路径 不以/开头 以当前资源的所在路径为出发点去找目标资源 语法: ./表示当前资源的路径 ../表示当前资源的上一层路径 缺点:不同位置,相对路径写法不同2.绝对路径 以固定的路径作为出发点作为目标资源,和当前资源所在路径没关系 语法:以/开头,不同的项目中,固定的路径…

当前2024阿里云服务器哪个地域价格比较优惠,哪个地域便宜?

目前2024年阿里云服务器地域对比哪个价格更优惠&#xff1f;华北6&#xff08;乌兰察布&#xff09;、华北3&#xff08;张家口&#xff09;、华北1&#xff08;青岛&#xff09;和华南2&#xff08;河源&#xff09;地域更便宜&#xff0c;云服务器吧yunfuwuqiba.com整理阿里云…

Python 网络请求:深入理解Requests库

目录 引言 一、Requests库简介 二、安装与基本使用 三、requests库的特性与优势 四、requests库在实际应用中的案例 1.get请求 2.post请求 3.超时重试 4.headers设置 5.session会话 6.携带cookie​​​​​​​ 7.携带代理​​​​​​​ 8.携带身份认证​​​​​…

SpringBoot学习笔记-S2

1. SpringBoot中的常见注解 RequestBody&#xff1a;使SpringMVC框架可自动读取请求体里面的JSON格式的数据&#xff0c;转换成map类型的集合对象RestController&#xff1a;开发RESTful API 时使用&#xff0c;等价于ResponseBody Controller。RestController和Controller的…

【简单讲解下WebSocket】

&#x1f308;个人主页:程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

卷积神经网络-池化层

卷积神经网络-池化层 池化层&#xff08;Pooling Layer&#xff09;是深度学习神经网络中的一个重要组成部分&#xff0c;通常用于减少特征图的空间尺寸&#xff0c;从而降低模型复杂度和计算量&#xff0c;同时还能增强模型的不变性和鲁棒性。 池化操作通常在卷积神经网络&am…

号卡极团分销管理系统 index.php SQL注入漏洞复现

0x01 产品简介 号卡极团分销管理系统,同步对接多平台,同步订单信息,支持敢探号一键上架,首页多套UI+商品下单页多套模板,订单查询支持实时物流信息、支持代理商自定义域名、泛域名绑定,内置敢探号、172平台、号氪云平台第三方接口以及号卡网同系统对接! 0x02 漏洞概述…

深入解析Hadoop生态核心组件:HDFS、MapReduce和YARN

这里写目录标题 01HDFS02Yarn03Hive04HBase1&#xff0e;特点2&#xff0e;存储 05Spark及Spark Streaming关于作者&#xff1a;推荐理由&#xff1a;作者直播推荐&#xff1a; 一篇讲明白 Hadoop 生态的三大部件 进入大数据阶段就意味着进入NoSQL阶段&#xff0c;更多的是面向…

如何用python绘制地图并计算地图上两点之间的距离【附源码】

一、这段代码用plotiy库绘制了可视化世界平面地图,计算了纽约和洛杉矶两个城市之间的距离,并将其打印输出。同时,在地图上显示了这两个城市的坐标点,具体代码如下: import plotly.graph_objects as go from geopy.distance import geodesic # 创建地图对象 fig = go.Fig…

Docker Desktop 不支持 host 网络模式

先把这个结论的放在前面&#xff0c;直接访问链接就能看到官方文档中已经明确说了不支持。 参考链接&#xff1a;docker desktop for windows 不支持 host 网络模式 以前对于 docker 的网络模式&#xff0c;一直只是了解&#xff0c;没有亲自尝试过。结果今天在尝试 docker 的 …

Android12 简单的共享内存驱动实现 参考Ashmem

Android12 共享内存驱动实现 SOC&#xff1a;RK3568 system&#xff1a;Android12 概述&#xff1a; 1. 概述 Ashmem&#xff08;Anonymous Shared Memory&#xff0c;Android 匿名共享内存&#xff09;&#xff0c;它基于 mmap 系统调用&#xff0c;可以让不同进程将同一段…

Flutter Web 的未来,Wasm Native 即将到来

早在去年 Google I/O 发布 Flutter 3.10 的时候就提到过&#xff0c; Flutter Web 的未来会是 Wasm Native &#xff0c;当时 Flutter 团队就表示&#xff0c;Flutter Web 的定位不是设计为通用 Web 的框架&#xff0c;类似的 Web 框架现在有很多&#xff0c;而 Flutter 的定位…

DIY蓝牙键盘(1) - 理解 键盘报文(免费)

DIY蓝牙键盘(1) - 理解键盘报文 1. 键盘报文体验 一个键盘对于用户的体验是&#xff0c;用户按按键A他能看到字母A会在主机上显示出来。那这是如何实现的&#xff1f; 其实很简单&#xff0c;只要键盘发送下面的两个报文给主机&#xff0c;字母A就能在主机上显示出来。 (1)…

如何打包一个手机软件

目录 前言&#xff1a; 准备工具&#xff1a; 创建项目&#xff1a; 打包程序&#xff1a; 前言&#xff1a; 我们平时手机上使用的程序&#xff0c;或者电脑上使用的程序都可以由Web程序打包而来的&#xff0c;而打包不是一个.html文件也不是一个.js文件而是一个大型的文…

Centos-Linux安装部署MySQL-8.0

linux搭建mysql 1.使用系统的root账户 2.切换到 /usr/local 目录下 3.下载mysql wget https://dev.mysql.com/get/Downloads/MySQL-8.0/mysql-8.0.31-linux-glibc2.12-x86_64.tar.xz也可以直接在官方下载最新版本 官方下载 选择linux 4.解压mysql tar xvJf mysql-8.0.21-l…

MySQL基础【语句执行顺序】

一个SQL语句它的执行顺序对于我们思考题意有着很重要的关系 题意就是&#xff1a;找出哪些只逛超市不买单的人&#xff08;买单0元也算哦&#xff0c;可能是使用的是代金券吧&#xff09; 看到此题关键找出两个数据 参观过的人 和 买单的人 他们的差就是白嫖的人&#xff08;支…

Excel、PowerQuery 和 ChatGPT 终极手册(上)

原文&#xff1a;Ultimate ChatGPT Handbook for Enterprises 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 序言 在不断发展的数据管理和分析领域中&#xff0c;掌握 Excel 的查找功能不仅是一种技能&#xff0c;更是高效数据处理的基石。《使用 Power Query 和 Ch…