GOF23种设计模式面试之三种工厂模式

2023-11-22 04:30

本文主要是介绍GOF23种设计模式面试之三种工厂模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

简单工厂

简单工厂模式属于创建型模式,但是并不属于GOF23种设计模式,通过简单工厂模式,我们可以了解一下工厂模式相关的概念。

将实例化的操作放在一个类中,这个类就成为简单工厂类,简单工厂模式,定义了某一个工厂对象能够创建出哪一种产品类的实例。

借此将客户类和具体子类的实现进行解耦,客户端只注重于传入工厂类的参数,对于创建对象的逻辑不关心


例:
现在有一个抽象类Sword,可以打造各种剑

public abstract class Sword {public abstract void produce();
}

它的子类有东方剑EastSword

public class EastSword extends Sword{@Overridepublic void produce() {System.out.println("produce a EastSword");}
}

西洋剑 WesternSword

public class WesternSword extends Sword {@Overridepublic void produce() {System.out.println("produce a WestrnSword");}
}

现在准备打造剑:

如果这样打造的话,打造10种不同的剑则需要依赖10个不同的类

public class Client {public static void main(String[] args) {//Sword sword = new WesternSword();Sword sword = new EastSword();sword.produce();}
}

所以我们利用简单工厂的方式,只需要传递参数,让工厂来为我们生成对象,只需要依赖工厂这个类

//简单工厂
public class SwordSimpleFactory {public static Sword getSword(Class c){Sword sword = null;try {sword = (Sword) Class.forName(c.getName()).newInstance();} catch (InstantiationException e) {e.printStackTrace();} catch (IllegalAccessException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}return sword;}
}
//对应的客户端
public class Client {public static void main(String[] args){//Sword sword = SwordSimpleFactory.getSword(WesternSword.class);Sword sword = SwordSimpleFactory.getSword(EastSword.class);sword.produce();}
}

优点:

  • 只需要传入正确的参数,来获取需要的对象,而不需要了解创建的细节

缺点:

  • 增加新的产品时,需要对工厂类的内部逻辑进行修改,违背了开闭原则

工厂方法

工厂方法定义了一个创建对象的接口,但由实现这个接口的子类决定要实例化哪个类,工厂方法把实例化操作推迟到子类。

例:
依旧是东方剑和西洋剑的例子

现在我们创建一个工厂的接口:SwordFactory,但是并不在这里实现创建对象的逻辑

public interface SwordFactory {Sword getSword();
}

创建对象的实现交给了该工厂的实现:EastFactory,WesternFactory

public class EastFactory implements SwordFactory{public Sword getSword() {return new EastSword();}
}public class WesternFactory implements SwordFactory{public Sword getSword() {return new WesternSword();}
}

通过对应的工厂方法来生成对象

public class Client {public static void main(String[] args){//SwordFactory swordFactory = new WesternFactory();SwordFactory swordFactory = new EastFactory();Sword sword = swordFactory.getSword();sword.produce();}
}

这样不用在工厂里生成对应的逻辑,而是使用子类工厂创建对应的对象,符合开闭原则。当需要拓展生成新的对象,就实现一个新的子类工厂。

优点:

  • 用户只需要关注产品对应的工厂,不用在意产品的创建细节
  • 对于新产品的加入符合了开闭原则,提高可扩展性

缺点:

  • 创建类的数量过多,会增加复杂度

抽象工厂

抽象工厂模式提供一个创建一系列相关或相互依赖对象的接口,抽象工厂模式注重的是一个对象家族(产品族)。

该模式强调了一系列相关的产品对象(属于同一产品族)的生成,从一个产品族工厂取出来的产品一定属于同一产品族


产品族和产品等级结构

什么是产品族和产品等级结构?

例如:

刀和剑,是不同的产品,而东方刀和东方剑属于东方这一产品族,西洋刀和西洋剑属于西方的产品族。

而东方刀和西洋刀都是刀,属于同一产品等级结构,同理,东方剑和西洋剑也是同一产品等级结构。

在这里插入图片描述
而抽象工厂关心的就是产品族,比如对于上面的紫色图形,可以创建一个紫色工厂,该工厂可以生产紫色三角形,紫色圆形,紫色五边形。

例:
继续上面的例子

现在有了东方的刀剑

public class EastKnife extends Knife{@Overridepublic void produce() {System.out.println("produce a EastKnife");}
}public class EastSword extends Sword {@Overridepublic void produce() {System.out.println("produce a EastSword");}
}

和西洋的刀剑

public class WesternKnife extends Knife{@Overridepublic void produce() {System.out.println("produce a WesternKnife");}
}public class WesternSword extends Sword {@Overridepublic void produce() {System.out.println("produce a WestrnSword");}
}

定义一个抽象工厂:WeaponFactory 可以生产刀剑

public interface WeaponFactory {Sword getSword();Knife getKnife();
}

所以就有了东方武器工厂和西方武器工厂

//东方武器工厂
public class EastWeaponFactory implements WeaponFactory{public Sword getSword() {return new EastSword();}public Knife getKnife() {return new EastKnife();}
}//西方武器工厂
public class WesternWeaponFactory implements WeaponFactory{public Sword getSword() {return new WesternSword();}public Knife getKnife() {return new WesternKnife();}
}

一个武器工厂,生产出来的产品一定是同一产品族的

public class Client {public static void main(String[] args){//WeaponFactory weaponFactory = new WesternWeaponFactory();WeaponFactory weaponFactory = new EastWeaponFactory();Sword sword = weaponFactory.getSword();Knife knife = weaponFactory.getKnife();sword.produce();	// produce a EastSwordknife.produce();	// produce a EastKnife}
}

优点

  • 将相关联的产品族统一到一个工厂内创建

缺点

  • 工厂逻辑编写时就规定了所有产品集合,当需要在产品族中扩展新的产品时,则需要修改抽象工厂的接口

这篇关于GOF23种设计模式面试之三种工厂模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

字节面试 | 如何测试RocketMQ、RocketMQ?

字节面试:RocketMQ是怎么测试的呢? 答: 首先保证消息的消费正确、设计逆向用例,在验证消息内容为空等情况时的消费正确性; 推送大批量MQ,通过Admin控制台查看MQ消费的情况,是否出现消费假死、TPS是否正常等等问题。(上述都是临场发挥,但是RocketMQ真正的测试点,还真的需要探讨) 01 先了解RocketMQ 作为测试也是要简单了解RocketMQ。简单来说,就是一个分

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

秋招最新大模型算法面试,熬夜都要肝完它

💥大家在面试大模型LLM这个板块的时候,不知道面试完会不会复盘、总结,做笔记的习惯,这份大模型算法岗面试八股笔记也帮助不少人拿到过offer ✨对于面试大模型算法工程师会有一定的帮助,都附有完整答案,熬夜也要看完,祝大家一臂之力 这份《大模型算法工程师面试题》已经上传CSDN,还有完整版的大模型 AI 学习资料,朋友们如果需要可以微信扫描下方CSDN官方认证二维码免费领取【保证100%免费

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

模版方法模式template method

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/template-method 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤。 上层接口有默认实现的方法和子类需要自己实现的方法

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

迭代器模式iterator

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/iterator 不暴露集合底层表现形式 (列表、 栈和树等) 的情况下遍历集合中所有的元素

《x86汇编语言:从实模式到保护模式》视频来了

《x86汇编语言:从实模式到保护模式》视频来了 很多朋友留言,说我的专栏《x86汇编语言:从实模式到保护模式》写得很详细,还有的朋友希望我能写得更细,最好是覆盖全书的所有章节。 毕竟我不是作者,只有作者的解读才是最权威的。 当初我学习这本书的时候,只能靠自己摸索,网上搜不到什么好资源。 如果你正在学这本书或者汇编语言,那你有福气了。 本书作者李忠老师,以此书为蓝本,录制了全套视频。 试

利用命令模式构建高效的手游后端架构

在现代手游开发中,后端架构的设计对于支持高并发、快速迭代和复杂游戏逻辑至关重要。命令模式作为一种行为设计模式,可以有效地解耦请求的发起者与接收者,提升系统的可维护性和扩展性。本文将深入探讨如何利用命令模式构建一个强大且灵活的手游后端架构。 1. 命令模式的概念与优势 命令模式通过将请求封装为对象,使得请求的发起者和接收者之间的耦合度降低。这种模式的主要优势包括: 解耦请求发起者与处理者

java面试常见问题之Hibernate总结

1  Hibernate的检索方式 Ø  导航对象图检索(根据已经加载的对象,导航到其他对象。) Ø  OID检索(按照对象的OID来检索对象。) Ø  HQL检索(使用面向对象的HQL查询语言。) Ø  QBC检索(使用QBC(Qurey By Criteria)API来检索对象。 QBC/QBE离线/在线) Ø  本地SQL检索(使用本地数据库的SQL查询语句。) 包括Hibern