探索设计模式的魅力:工厂方法模式

工厂方法模式是一种创建型设计模式,它提供了一种创建对象的接口,但将具体实例化对象的工作推迟到子类中完成。这样做的目的是创建对象时不用依赖于具体的类,而是依赖于抽象,这提高了系统的灵活性和可扩展性。

以下是工厂方法模式的几个关键组成部分:

  1. 产品(Product): 定义了工厂方法所创建的对象的接口。在我们的日志记录器示例中,Logger 类就是一个产品接口。

  2. 具体产品(Concrete Product): 实现了产品接口的具体类。继承自Logger类的FileLoggerConsoleLogger 和 DatabaseLogger类就是具体产品。

  3. 创建者(Creator): 声明了工厂方法,这个方法返回一个产品类型的对象。通常情况下,创建者类将是抽象类,并包含工厂方法的声明。在我们的示例中,LoggerFactory就是一个创建者。

  4. 具体创建者(Concrete Creator): 覆盖了工厂方法以返回一个具体产品实例。这是实际决定要实例化哪一个产品的类。例如,FileLoggerFactoryConsoleLoggerFactory 和 DatabaseLoggerFactory都是具体创建者,它们覆盖工厂方法以返回它们各自的产品实例。

工厂方法模式的工作方式:

  • 定义产品接口: 首先定义一个产品接口,它描述了产品的公共接口。
  • 创建具体产品: 然后为每种类型的产品实现具体类。
  • 创建抽象创建者: 接着创建一个创建者(通常是抽象类或接口)来声明工厂方法。工厂方法通常会有一个返回类型为产品接口的返回类型。
  • 实现具体创建者: 创建具体创建者类来实现抽象创建者中声明的工厂方法,返回具体产品的实例。
  • 在应用中使用创建者: 最后在应用程序中,我们使用创建者类来调用工厂方法,获取产品对象的实例。

工厂方法模式的优点:

  • 降低耦合度: 客户代码从具体类解耦,并依赖于抽象。这意味着客户代码不需要改变就能与任何新增的具体产品工作。
  • 增加了系统的可扩展性: 新的具体产品可以很容易地加入到系统中,因为现有的客户代码不会受到影响。
  • 提高代码的可维护性: 如果一种产品在多处创建,更改产品的实现或者更换一个产品都会更加容易和集中。

工厂方法模式的缺点:

  • 增加了代码的复杂性: 可能需要引入许多新类,每种类型的产品都需要一个新的具体创建者类。
  • 需要更多的设计考虑: 设计好抽象创建者和具体创建者之间的关系需要一定的设计经验和考虑。

总的来说,工厂方法模式在需要灵活和可扩展的系统中非常有用,尤其是当我们预计产品类可能会经常改变时。它有助于保持一个健壮而灵活的代码库。

目录

一、案例

1.1 示例代码

1.2 扩展-添加数据库日志

二、模式讲解

2.1 功能

2.2 工厂方法模式结构

2.3 示例代码程序结构图

2.4 简单工厂方法结构图

2.5 工厂方法模式与简单工厂模式

三、扩展-工厂方法与IoC/DI

  3.1 弄明白IoC/DI

  3.2 工厂方法与IoC/DI的关系


一、案例

场景

        需要一个创建不同类型日志记录器的框架,日志记录器可能记录到文件、控制台或者数据库。

1.1 示例代码

        定义一个抽象日志记录器类和一个工厂方法:

/*** 日志处理抽象类*/
public abstract class Logger {/*** 操作日志*/public abstract void log(String message);
}/*** 工厂方法抽象类*/
public abstract class LoggerFactory {/*** 操作日志*/public void log(String message) {createLogger().log(message);}/*** 工厂方法,创建日志对象的接口对象*/public abstract Logger createLogger();
}

        为文件日志类型实现具体的日志记录器和对应的工厂:

/*** 文件日志实现类*/
public class FileLogger extends Logger {public void log(String message) {// 逻辑来将消息写入文件System.out.println("File logger: " + message);}
}/*** 文件日志工厂方法类*/
public class FileLoggerFactory extends LoggerFactory {@Overridepublic Logger createLogger() {// 可以在这里添加创建FileLogger所需的逻辑和初始化return new FileLogger();}
}

        为控制台日志类型实现具体的日志记录器和对应的工厂:

/*** 控制台日志实现类*/
public class ConsoleLogger extends Logger {public void log(String message) {// 逻辑来将消息打印到控制台System.out.println("Console logger: " + message);}
}/*** 控制台工厂方法类*/
public class ConsoleLoggerFactory extends LoggerFactory {@Overridepublic Logger createLogger() {// 创建ConsoleLogger的逻辑return new ConsoleLogger();}
}

        现在,在应用程序中,我们可以根据需要使用工厂来创建日志记录器对象,而不必直接实例化它们。这样,如果以后需要添加新的日志记录器类型(例如,数据库日志记录器),我们只需要添加一个新的工厂而不需要修改现有代码。

        客户端示例代码:

public class Application {public static void main(String[] args) {LoggerFactory factory = new FileLoggerFactory();// 或者 factory = new ConsoleLoggerFactory();factory.log("这是一条日志信息.");}
}

        当 Application 运行时,根据选择的工厂类型,它将使用对应的工厂创建一个日志记录器,并通过这个记录器记录消息。这个示例遵循了工厂方法的设计原则,因为它使对象的创建和使用分离,使得系统易于扩展和维护。

        

1.2 扩展-添加数据库日志

        如果我们想要将日志记录扩展到数据库,我们首先需要为数据库日志创建一个新的Logger子类,然后实现对应的工厂类。下面展示了如何实现这一扩展:

// 数据库日志记录器——实现Logger抽象类
public class DatabaseLogger extends Logger {public void log(String message) {// 示例逻辑来将消息保存到数据库System.out.println("Database logger: " + message);// 这里可以包含实际将日志保存到数据库的代码}
}// 数据库日志工厂——继承LoggerFactory
public class DatabaseLoggerFactory extends LoggerFactory {@Overridepublic Logger createLogger() {// 创建DatabaseLogger的逻辑,可以在这里包含初始化代码return new DatabaseLogger();}
}

        在Application 或其他任何需要日志记录功能的部分,现在可以不作出太多改动地简单地引入新的DatabaseLogger:

public class Application {public static void main(String[] args) {LoggerFactory factory = new DatabaseLoggerFactory();// 或者 factory = new FileLoggerFactory();// 或者 factory = new ConsoleLoggerFactory();factory.log("这是一条日志信息.");}
}

        现在,无论是FileLoggerFactoryConsoleLoggerFactory还是DatabaseLoggerFactory,均不需要修改 Application 中的任何代码。进一步说,如果有必要添加其他类型的日志记录,如远程API日志记录、XML日志记录等,整个流程同样适用。这就展示了工厂方法模式在扩展性方面的强大之处。每次新增一种产品(本例中的Logger实现),只需添加一个新的具体工厂类且不需要改动现有的代码,符合开闭原则(对修改封闭,对扩展开放)。

        通过上述示例,你可以看到工厂方法设计模式是如何工作的,它能够提供足够的灵活性,允许系统在不直接依赖具体类的情况下创建对象。这种方式降低了类间的耦合,提高了代码的可维护性与可扩展性。

        

二、模式讲解

2.1 功能

功能工厂方法主要工能是让父类在不知道具体实现的情况下,完成自身的功能调用;而具体的实现延迟到子类来实现

这样在设计的时候,不用去考虑具体的实现,需要某个对象,把它通过工厂方法返回就好了, 在使用这些对象实现功能的时候还是通过接口来操作,这类似于 IoC/DI 的思想。

        

2.2 工厂方法模式结构

  • Logger:定义工厂方法所创建的对象的类,也就是实际需要使用的对象的类。
  • 子类A:具体的 Logger 接口的实现对象。
  • Factory:创建器,声明工厂方法,工厂方法通常会返回一个 Logger 类型的实例对象,而且多是抽象方法。也可以在 Factory里面提供工厂方法的默认实现,让工厂方法返回一个缺省的Logger类型的实现对象。
  • 实现类B:具体的创建器对象,覆盖实现 Factory 定义的工厂方法,返回具体的 Logger实例。

        

2.3 示例代码程序结构图

         

2.4 简单工厂方法结构图

         

2.5 工厂方法模式与简单工厂模式

        工厂方法模式与简单工厂模式结构如上图2 和 图3。

        简单工厂模式见:探索设计模式的魅力:简单工厂模式-CSDN博客文章浏览阅读1.5k次,点赞50次,收藏36次。实现简单工厂的难点就在于 “如何选择” 实现,前面便子中传递参数的方法, 那都是静态的参数,还可以实现成为动态的参数。客户端通过简单工厂创建 了一个实现接口的对象,然后面向接口编程,从客户端来看,它根本不知道具体的实现是什么,也不知道是如何实现的,它只知道通过工厂获得了一个接口对象 , 然后通过这个接口来获取想要的功能。如果通过客户端的参数来选择具体的实现类,那么就必须让客户端能理解各个参数所代表的具体功能和含义,这样会增加客户端使用的难度,也部分暴露了内部实现,这种情况可以选用可配置的方式来实现。https://blog.csdn.net/danci_/article/details/135566105        若要添加新的日志类型,简单工厂模式需要添加这个新功能类的日志子类,再在Factory中添加一个case语句来做判断;工厂方法模式需要添加这个新功能类的日志子类,再添加一个实现Factory的工厂类。但要我再去更改客户端,这 不等于不但没有减化难度,反而增加了很多类和方法,把复杂性增加了 吗?为什么要这样?”

        简单工厂模式最大优点在于工厂类中包含了必要的逻辑判断,根据客户端的选择条件动态实例化相关的类,对于客户端来说,去除了与具体产品的依赖。但是如果添加新的日志类型,就要去个性case判断条件,这违背了开-闭原则。(工厂类扩展了,也修改了)

        于是工厂方法出现了。

工厂方法模式(Factory Method):定义一个用于创建对象的接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到其子类。

        此时,添加新的日志类型,只需要添加新的日志子炻和新的日志工厂类,满足了开-闭原则。(对扩展开放,对修改关闭)

        

三、扩展-工厂方法与IoC/DI

        IOC-Inversion of Control,控制反转。

        DI-Dependency Injection,依赖流入。

  3.1 弄明白IoC/DI

  1. 参与者:一般有三方参与者,一个是某个对象;另一个是IoC/DI 的容器;还有一个是某个对象的外部资源。

  2. 谁依赖于谁: 当然是某个对象依赖于 IoC/DI 的容器。

  3. 为什么需要依赖:对象需要 IoC/DI 的容器来提供对象需要的外部资源。

  4. 谁注入于谁:很明 显是 IoC/DI 的容器注入某个对象。

  5. 到底注入什么:就是注入某个对象所需要的外部资源。

  6. 谁控制谁:当然是 IoC/DI 的容器来控制对象了。

  7. 控制什么:主要是控制对象实例的创建。

  8. 为何叫反转 : 反转是相对于正向而言的,那么什么算是正向的呢? 考虑一下常规情况下的应用程序,如果要在A 里面使用C,你会怎么做呢? 当然是直接去创建C 的对象,也就是说,在A 类中主动去获取所需要的外部资源C,这种情况被称为正向的。

        那么什么是反向呢?就是A 类不再主动去获取C,而是被动等待,等待IoC/DI 的容器获 取一个C的实例,然后反向地注入到A 类中。

  9. 依赖注入和控制反转是同一概念吗?

        依赖注入和控制反转是对同一件事情的不同描述。 从某个方面讲,就是它们描述的角度不同。依赖注入是从应用程序的角度去描述,可以 把依赖注入描述得完整点 :应用程序依赖容器创建并注入它所需要的外部资源;而控制 反转是从容器的角度去描述,描述得完整点就是 :容器控制应用程序,由容器反向地向 应用程序注入其所需要的外部资源。

        小结:其实 IoC/DI 对编程带来的最大改变不是在代码 上,而是在思想上,发生了“主从换位”的变化。应用程序原本是老大,要获取什么资源都是主动出击, 但是在 IoC/DI 思想中,应用程序就变成被动的了,被动地等待 IoC/DI 容器来创建并注入它所需要的资源了。

        

  3.2 工厂方法与IoC/DI的关系

        IoC/ DI:主从换位,被动等待IOC/DI容器来创建并流入它所需要的资源。

        工厂方法

/*** 工厂方法抽象类*/
public abstract class LoggerFactory {/*** 操作日志*/public void log(String message) {createLogger().log(message);}/*** 工厂方法,创建日志对象的接口对象*/public abstract Logger createLogger();
}

        log方法中,需要用Logger类,可是又不知道要用哪一个,也不去主动去创建了,反正在子类里已经实现了,不用管怎么获取,直接使用日志功能,类似于从子类注入进来。   

        从思想层面上,会发现工厂方法示模式和 IoC/DI 的思想是相似的,都是“ 主动变被动” ,进行了“ 主从换位” ,从而获得了更灵活的程序结构。

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

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

相关文章

学习视频一些杂乱的东西

文章目录 ref获取dom元素监听深层的某个属性? 可选链操作符 和 ?? 双问号表达式v-slot 语法糖作用域插槽动态插槽 初始化数组骚操作数字滚动 -> gsapstyle妙招新奇的原型链 object.createB站笔记链接JS相关设计模式ajaxsvgvue3scsswebpack内存泄漏 ref获取dom元素 直接给…

基于深度学习的实例分割的Web应用

基于深度学习的实例分割的Web应用 1. 项目简介1.1 模型部署1.2 Web应用 2. Web前端开发3. Web后端开发4. 总结 1. 项目简介 这是一个基于深度学习的实例分割Web应用的项目介绍。该项目使用PaddlePaddle框架,并以PaddleSeg训练的图像分割模型为例。 1.1 模型部署 …

【iOS】数据存储方式总结(持久化)沙盒结构

在iOS开发中,我们经常性地需要存储一些状态和数据,比如用户对于App的相关设置、需要在本地缓存的数据等等,本篇文章将介绍六个主要的数据存储方式 iOS中数据存储方式(数据持久化) 根据要存储的数据大小、存储数据以及…

案例:应用内字体大小调节

文章目录 介绍相关概念完整实例 代码结构解读保存默认大小获取字体大小修改字体大小 介绍 本篇Codelab将介绍如何使用基础组件Slider,通过拖动滑块调节应用内字体大小。要求完成以下功能: 实现两个页面的UX:主页面和字体大小调节页面。拖动…

基于物联网设计的智能储物柜(4G+华为云IOT+微信小程序)

一、项目介绍 在游乐场、商场、景区等人流量较大的地方,往往存在用户需要临时存放物品的情况,例如行李箱、外套、购物袋等。为了满足用户的储物需求,并提供更加便捷的服务体验,当前设计了一款物联网智能储物柜。 该智能储物柜通…

git提交报错:remote: Please remove the file from history and try again.

1. 报错信息 remote: error: File: fba7046b22fd74b77425aa3e4eae0ea992d44998 500.28 MB, exceeds 100.00 MB. remote: Please remove the file from history and try again. git rev-list --objects --all | grep fba7046b22fd74b77425aa3e4eae0ea992d44998 2. 分析原因 e…

打架识别摄像机

随着社会治安问题的增加,打架事件在公共场所频繁发生,给社会治安带来了一定程度的威胁。因此,为了提高公共场所的安全性,可以利用现代科技,如人工智能和摄像技术,开发一种打架识别摄像机。 这种摄像机可以通…

基于 IDEA 进行 Maven 工程构建

一、构建概念和构建过程 项目构建是指将源代码、依赖库和资源文件等转换成可执行或可部署的应用程序的过程,在这个过程中包括编译源代码、链接依赖库、打包和部署等多个步骤。 项目构建是软件开发过程中至关重要的一部分,它能够大大提高软件开发效率&…

【Docker】CentOS stream 上安装 Docker 环境详细指南

文章目录 1. 定义2. 优势3. 安装1)Linux 上安装(强烈推荐)2)Windows 和 MAC 上安装 4. 验证1)查看版本2)运行 Hello World 总结 Docker 是一种轻量级的容器化技术,提供了一种在不同环境中快速、…

Maven普通工程和web工程创建

文章目录 创建项目前设置maven工程前设置工作创建项目前--》设置utf-8配置maven参数Maven普通工程和web工程创建Maven简单工程第一步:File–New–Project 第二步:选择maven然后下一步:填写后询选择finish初始化maven工程目录简介maven简单工程…

Rust-借用检查

Rust语言的核心特点是:在没有放弃对内存的直接控制力的情况下,实现了内存安全。 所谓对内存的直接控制能力,前文已经有所展示:可以自行决定内存布局,包括在栈上分配内存,还是在堆上分配内存;支…

使用vue快速开发一个带弹窗的Chrome插件

vue-chrome-extension-quickstart 说在前面 🎈平时我们使用Chrome插件通常都只是用来编写简单的js注入脚本,大家有没有遇到过需要插件在页面上注入一个弹窗呢?比如我们希望可以通过快捷键快速唤起ChatGPT面板或者快速唤起一个翻译面板&#x…

自动化革命:大象机器人的Mercury A1机械臂

引言 大象机器人的Mercury系列,是面向工业自动化和智能制造的新型机械臂产品线。这些机械臂不仅在设计上创新,还在材料选择上使用了碳纤维、铝合金和工程塑料等轻质强韧材料,搭载高精度谐波减速器。Mercury系列的推出,反映了大象机…

day2:TCP、UDP网络通信模型

思维导图 机械臂实现 #include <head.h> #define SER_POTR 8899 #define SER_IP "192.168.125.223" int main(int argc, const char *argv[]) {//创建套接字int cfdsocket(AF_INET,SOCK_STREAM,0);if(cfd-1){perror("");return -1;}//链接struct so…

部署MinIO

一、安装部署MINIO 1.1 下载 wget https://dl.min.io/server/minio/release/linux-arm64/minio chmod x minio mv minio /usr/local/bin/ # 控制台启动可参考如下命令, 守护进程启动请看下一个代码块 # ./minio server /data /data --console-address ":9001"1.2 配…

深度学习笔记(七)——基于Iris/MNIST数据集构建基础的分类网络算法实战

文中程序以Tensorflow-2.6.0为例 部分概念包含笔者个人理解&#xff0c;如有遗漏或错误&#xff0c;欢迎评论或私信指正。 截图和程序部分引用自北京大学机器学习公开课 认识网络的构建结构 在神经网络的构建过程中&#xff0c;都避不开以下几个步骤&#xff1a; 导入网络和依…

【Android+物联网】Android封装MQTT连接阿里云物联网平台

前言&#xff1a; 亲测可行&#xff0c;本文实现Android封装MQTT连接阿里云物联网平台。将MQTT协议和连接阿里云平台的操作通过Android studio写入APP中&#xff0c;并简单设计UI。实现手机APP远程控制单片机LED灯亮灭的功能。 关于《Android软件开发》&#xff0c;见如下专栏…

手拉手Vue3生命周期实战应用

每个 Vue 组件实例在创建时都需要经历一系列的初始化步骤&#xff0c;比如设置好数据侦听&#xff0c;编译模板&#xff0c;挂载实例到 DOM&#xff0c;以及在数据改变时更新 DOM。在此过程中&#xff0c;它也会运行被称为生命周期钩子的函数&#xff0c;让开发者有机会在特定阶…

SpringAMQP的使用

1. 简介&#xff1a; SpringAMQP是基于RabbitMQ封装的一套模板&#xff0c;并且还利用SpringBoot对其实现了自动装配&#xff0c;使用起来非常方便。 SpringAmqp的官方地址&#xff1a;https://spring.io/projects/spring-amqp SpringAMQP提供了三个功能&#xff1a; 自动声…

【linux】查看Debian应用程序图标对应的可执行命令

在Debian系统中&#xff0c;应用程序图标通常与.desktop文件关联。您可以通过查看.desktop文件来找到对应的可执行命令。这些文件通常位于/usr/share/applications/或~/.local/share/applications/目录下。这里是如何查找的步骤&#xff1a; 1. 打开文件管理器或终端。 2. 导…