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

相关文章

Java图片压缩三种高效压缩方案详细解析

《Java图片压缩三种高效压缩方案详细解析》图片压缩通常涉及减少图片的尺寸缩放、调整图片的质量(针对JPEG、PNG等)、使用特定的算法来减少图片的数据量等,:本文主要介绍Java图片压缩三种高效... 目录一、基于OpenCV的智能尺寸压缩技术亮点:适用场景:二、JPEG质量参数压缩关键技术:压缩效果对比

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

SpringBoot如何通过Map实现策略模式

《SpringBoot如何通过Map实现策略模式》策略模式是一种行为设计模式,它允许在运行时选择算法的行为,在Spring框架中,我们可以利用@Resource注解和Map集合来优雅地实现策略模式,这... 目录前言底层机制解析Spring的集合类型自动装配@Resource注解的行为实现原理使用直接使用M

C语言实现两个变量值交换的三种方式

《C语言实现两个变量值交换的三种方式》两个变量值的交换是编程中最常见的问题之一,以下将介绍三种变量的交换方式,其中第一种方式是最常用也是最实用的,后两种方式一般只在特殊限制下使用,需要的朋友可以参考下... 目录1.使用临时变量(推荐)2.相加和相减的方式(值较大时可能丢失数据)3.按位异或运算1.使用临时

Java终止正在运行的线程的三种方法

《Java终止正在运行的线程的三种方法》停止一个线程意味着在任务处理完任务之前停掉正在做的操作,也就是放弃当前的操作,停止一个线程可以用Thread.stop()方法,但最好不要用它,本文给大家介绍了... 目录前言1. 停止不了的线程2. 判断线程是否停止状态3. 能停止的线程–异常法4. 在沉睡中停止5

Linux配置IP地址的三种实现方式

《Linux配置IP地址的三种实现方式》:本文主要介绍Linux配置IP地址的三种实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录环境RedHat9第一种安装 直接配置网卡文件第二种方式 nmcli(Networkmanager command-line

C#原型模式之如何通过克隆对象来优化创建过程

《C#原型模式之如何通过克隆对象来优化创建过程》原型模式是一种创建型设计模式,通过克隆现有对象来创建新对象,避免重复的创建成本和复杂的初始化过程,它适用于对象创建过程复杂、需要大量相似对象或避免重复初... 目录什么是原型模式?原型模式的工作原理C#中如何实现原型模式?1. 定义原型接口2. 实现原型接口3

大数据spark3.5安装部署之local模式详解

《大数据spark3.5安装部署之local模式详解》本文介绍了如何在本地模式下安装和配置Spark,并展示了如何使用SparkShell进行基本的数据处理操作,同时,还介绍了如何通过Spark-su... 目录下载上传解压配置jdk解压配置环境变量启动查看交互操作命令行提交应用spark,一个数据处理框架

Linux下修改hostname的三种实现方式

《Linux下修改hostname的三种实现方式》:本文主要介绍Linux下修改hostname的三种实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux下修改ho编程stname三种方式方法1:修改配置文件方法2:hFvEWEostnamectl命

Go语言中三种容器类型的数据结构详解

《Go语言中三种容器类型的数据结构详解》在Go语言中,有三种主要的容器类型用于存储和操作集合数据:本文主要介绍三者的使用与区别,感兴趣的小伙伴可以跟随小编一起学习一下... 目录基本概念1. 数组(Array)2. 切片(Slice)3. 映射(Map)对比总结注意事项基本概念在 Go 语言中,有三种主要