当前位置: 首页 > news >正文

设计模式:工厂方法模式(C#、JAVA、JavaScript、C++、Python、Go、PHP):

本节主要介绍设计模式中的工厂方法模式。

简介:

工厂方法模式,它是对简单工厂模式的进一步抽象化,其好处是可以使系统在不修改原来代码的情况下引进新的产品,即满足开闭原则。
它定义了一个用于创建对象的工厂接口,让子类决定实例化哪个类。在工厂方法模式中,父类负责定义创建对象的公共接口,而子类则负责生成具体的对象。这种模式将类实例化操作延迟到子类中完成,即由子类来决定究竟应该实例化哪个类。

工厂方法模式的创建步骤如下:
1、创建抽象工厂类,定义具体工厂的公共接口。
2、创建抽象产品类,定义具体产品的公共接口。
3、创建具体产品类,继承抽象产品类并定义具体产品的生产。
4、创建具体工厂类,继承抽象工厂类并定义创建对应具体产品实例的方法。
5、外界通过调用具体工厂类的方法,从而创建不同具体产品类的实例。

工厂方法模式的优点,主要包括:
1、增加新的产品类时无需修改现有系统:当需要增加一个新的产品时,只需要创建一个新的具体工厂和具体产品类,符合“开放-封闭”原则,增加了系统的灵活性和可扩展性。
2、封装了产品对象的创建细节:客户端只需要使用具体工厂类创建产品对象,无需关心对象是如何被创建的,这样就可以将产品对象的创建细节封装在具体工厂类中。
3、系统具有良好的灵活性和可扩展性:通过使用工厂方法模式,可以在不改变现有客户端代码的情况下,增加或修改产品类和工厂类,具有较强的灵活性。

工厂方法模式的缺点,主要包括:
1、增加额外的编写工作量:在增加新产品时,需要编写新的具体产品类和对应的具体工厂类,增加了系统的复杂度,需要更多的类需要编译和运行,给系统带来一些额外的开销。
2、需要考虑系统的抽象性和理解难度:为了增加系统的可扩展性,需要引入抽象层,在客户端代码中均使用抽象层进行定义,增加了系统的抽象性和理解难度,且在实现时可能需要用到DOM、反射等技术,增加了系统的实现难度。

​​​​​​​

示例

一、C#工厂方法模式

以下是一个示例,展示了如何在C#中实现工厂方法模式:

// 产品的抽象类  
public abstract class Product  
{  public abstract void Use();  
}  // 具体产品类1  
public class ConcreteProduct1 : Product  
{  public override void Use()  {  Console.WriteLine("使用具体产品1");  }  
}  // 具体产品类2  
public class ConcreteProduct2 : Product  
{  public override void Use()  {  Console.WriteLine("使用具体产品2");  }  
}  // 工厂的抽象类  
public abstract class Creator  
{  // 工厂方法,由子类实现具体的创建逻辑  public abstract Product CreateProduct();  
}  // 具体工厂类1  
public class ConcreteCreator1 : Creator  
{  public override Product CreateProduct()  {  return new ConcreteProduct1();  }  
}  // 具体工厂类2  
public class ConcreteCreator2 : Creator  
{  public override Product CreateProduct()  {  return new ConcreteProduct2();  }  
}  class Program  
{  static void Main(string[] args)  {  Creator creator1 = new ConcreteCreator1(); Product product1 = creator1.CreateProduct();   product1.Use();  Creator creator2 = new ConcreteCreator2  Product product2 = creator2.CreateProduct(); product2.Use();  }  
}

二、java工厂方法模式

以下是一个示例,展示了如何在Java中实现工厂方法模式:

// 抽象产品类  
public abstract class Product {  public abstract void use();  
}  // 具体产品类1  
public class ConcreteProduct1 extends Product {  @Override  public void use() {  System.out.println("使用具体产品1");  }  
}  // 具体产品类2  
public class ConcreteProduct2 extends Product {  @Override  public void use() {  System.out.println("使用具体产品2");  }  
}  // 抽象工厂类  
public abstract class Factory {  // 工厂方法,由子类实现具体的创建逻辑  public abstract Product createProduct();  
}  // 具体工厂类1  
public class ConcreteFactory1 extends Factory {  @Override  public Product createProduct() {  return new ConcreteProduct1();  }  
}  // 具体工厂类2  
public class ConcreteFactory2 extends Factory {  @Override  public Product createProduct() {  return new ConcreteProduct2();  }  
}  // Client代码  
public class Client {  public static void main(String[] args) {  Factory factory = new ConcreteFactory1(); // 可以根据实际需要更换为ConcreteFactory2  Product product = factory.createProduct();  product.use();  }  
}


三、javascript工厂方法模式

在JavaScript中,工厂方法模式通常可以通过构造函数和对象字面量的组合来实现。

// 抽象产品类  
class Product {  // 抽象方法  use() {  throw new Error('Use abstract method "use"');  }  
}  // 具体产品类1  
class ConcreteProduct1 extends Product {  use() {  console.log('使用具体产品1');  }  
}  // 具体产品类2  
class ConcreteProduct2 extends Product {  use() {  console.log('使用具体产品2');  }  
}  // 抽象工厂类  
class Factory {  // 工厂方法,由子类实现具体的创建逻辑  createProduct() {  throw new Error('Use abstract method "createProduct"');  }  
}  // 具体工厂类1  
class ConcreteFactory1 extends Factory {  createProduct() {  return new ConcreteProduct1();  }  
}  // 具体工厂类2  
class ConcreteFactory2 extends Factory {  createProduct() {  return new ConcreteProduct2();  }  
}  // Client代码  
class Client {  constructor(factory) {  this.factory = factory;  }  useProduct() {  let product = this.factory.createProduct();  product.use();  }  
}  // 使用Client类和ConcreteFactory1实例化一个新的Client对象并使用产品  
let client1 = new Client(new ConcreteFactory1());  
client1.useProduct();

四、C++工厂方法模式

以下是在C++中实现工厂方法模式:

//定义一个抽象产品类,它包含产品对象的公共接口。
class Product {  
public:  virtual void use() = 0; // 纯虚函数,具体实现由子类来决定  
};
//创建具体产品类,它们扩展了抽象产品类并实现了产品的具体行为。
class ConcreteProduct1 : public Product {  
public:  void use() override {  // 具体实现逻辑  std::cout << "使用具体产品1" << std::endl;  }  
};  class ConcreteProduct2 : public Product {  
public:  void use() override {  // 具体实现逻辑  std::cout << "使用具体产品2" << std::endl;  }  
};
//定义一个抽象工厂类,它包含一个工厂方法用于创建产品对象。这个方法是纯虚函数,具体实现由子类来决定。
class Factory {  
public:  virtual Product* createProduct() = 0; // 纯虚函数,具体实现由子类来决定  
};
//创建具体工厂类,它们扩展了抽象工厂类并实现了工厂方法的特定实现,以创建特定类型的产品对象。
class ConcreteFactory1 : public Factory {  
public:  Product* createProduct() override {  return new ConcreteProduct1();  }  
};  class ConcreteFactory2 : public Factory {  
public:  Product* createProduct() override {  return new ConcreteProduct2();  }  
};
//最后,在客户端代码中使用工厂方法模式来创建产品对象。客户端通过调用工厂对象的 createProduct 方法来创建产品对象,而不需要直接了解如何创建这些对象。这样可以提高客户端代码的灵活性和可维护性。
int main() {  Factory* factory = new ConcreteFactory1(); // 创建具体工厂对象  Product* product = factory->createProduct(); // 创建具体产品对象  product->use(); // 使用具体产品对象  delete factory; // 释放工厂对象内存  delete product; // 释放产品对象内存  return 0;  
}

五、python工厂方法模式

以下是在python中实现工厂方法模式:

from abc import ABCMeta, abstractmethod  # 抽象产品类  
class Product(metaclass=ABCMeta):  @abstractmethod  def operation(self):  pass  # 具体产品类1  
class ConcreteProduct1(Product):  def operation(self):  print("具体产品1被使用了")  # 具体产品类2  
class ConcreteProduct2(Product):  def operation(self):  print("具体产品2被使用了")  # 抽象工厂类  
class Factory(metaclass=ABCMeta):  @abstractmethod  def create_product(self):  pass  # 具体工厂类1  
class ConcreteFactory1(Factory):  def create_product(self):  return ConcreteProduct1()  # 具体工厂类2  
class ConcreteFactory2(Factory):  def create_product(self):  return ConcreteProduct2()  # Client代码  
if __name__ == "__main__":  factory1 = ConcreteFactory1()  product1 = factory1.create_product()  product1.operation()  factory2 = ConcreteFactory2()  product2 = factory2.create_product()  product2.operation()

    

六、go工厂方法模式

以下是一个示例,展示了如何在go中实现工厂方法模式:

//首先定义一个产品接口,该接口定义了产品的通用方法:
type Product interface {  Use()  
}
//然后,定义两个具体产品结构体,并实现Product接口的方法:
type ConcreteProduct1 struct{}  func (p *ConcreteProduct1) Use() {  fmt.Println("使用具体产品1")  
}  type ConcreteProduct2 struct{}  func (p *ConcreteProduct2) Use() {  fmt.Println("使用具体产品2")  
}
//接下来,定义一个工厂接口,该接口定义了一个创建产品的 方法:
type Factory interface {  CreateProduct() Product  
}
//然后,定义两个具体工厂结构体,并实现Factory接口的方法:
type ConcreteFactory1 struct{}  func (f *ConcreteFactory1) CreateProduct() Product {  return &ConcreteProduct1{}  
}  type ConcreteFactory2 struct{}  func (f *ConcreteFactory2) CreateProduct() Product {  return &ConcreteProduct2{}  
}
//最后,在客户端代码中,根据需要选择具体的工厂结构体实例化,然后使用该工厂结构体创建并使用产品:
func main() {  factory1 := &ConcreteFactory1{}  product1 := factory1.CreateProduct()  product1.Use()  factory2 := &ConcreteFactory2{}  product2 := factory2.CreateProduct()  product2.Use()  
}

七、PHP工厂方法模式

以下是一个示例,展示了如何在PHP中实现工厂方法模式:

//定义一个抽象产品接口(Abstract Product):
interface Product {  public function operation();  
}
//创建具体产品类实现抽象产品接口:
class ConcreteProduct1 implements Product {  public function operation() {  echo "具体产品1被使用了";  }  
}  class ConcreteProduct2 implements Product {  public function operation() {  echo "具体产品2被使用了";  }  
}
//定义一个抽象工厂类:
abstract class Creator {  abstract public function factoryMethod(): Product;  
}
//创建具体工厂类继承抽象工厂类:
class ConcreteCreator1 extends Creator {  public function factoryMethod() {  return new ConcreteProduct1();  }  
}  class ConcreteCreator2 extends Creator {  public function factoryMethod() {  return new ConcreteProduct2();  }  
}
//在客户端代码中,根据需要选择具体的工厂类实例化,并使用该工厂类创建并使用产品:
$creator1 = new ConcreteCreator1();  
$product1 = $creator1->factoryMethod();  
$product1->operation(); // 输出:具体产品1被使用了  $creator2 = new ConcreteCreator2();  
$product2 = $creator2->factoryMethod();  
$product2->operation(); // 输出:具体产品2被使用了

《完结》

相关文章:

设计模式:工厂方法模式(C#、JAVA、JavaScript、C++、Python、Go、PHP):

本节主要介绍设计模式中的工厂方法模式。 简介&#xff1a; 工厂方法模式&#xff0c;它是对简单工厂模式的进一步抽象化&#xff0c;其好处是可以使系统在不修改原来代码的情况下引进新的产品&#xff0c;即满足开闭原则。 它定义了一个用于创建对象的工厂接口&#xff0c;让…...

C++基础——指针

1 概述 指针的作用&#xff1a;可以通过指针间接访问内存 内存编号从0开始&#xff0c;一般使用十六进制数字表示&#xff0c;指针可以保存地址 2 指针变量定义和作用 int main() {//1、指针的定义int a 10; //定义整型变量a//指针定义语法&#xff1a; 数据类型 * 变量名 …...

PLC 学习day02 硬件输入/输入的知识

1.资料来源 1.链接&#xff1a;三菱PLC视频教程全集之FX3U基本单元输入接线_哔哩哔哩_bilibili 2. 链接&#xff1a; 三菱plc视频教程全集之FX3U基本单元输出接线_哔哩哔哩_bilibili 2. PLC 的输入部分器件连接。 2.1 PLC输入部分的硬件知识 1. 一般输入部分是PLC获取信息的地…...

rabbitMq (2)

RabbitMQ 消息应答与发布 文章目录 1. 消息应答1.2 自动应答1.2 手动应答1.3 代码案例 2. RabbitMQ 持久化2.1 队列持久化2.2 消息持久化 3. 不公平分发4. 预取值分发5. 发布确认5.1 发布确认逻辑5.2 开启发布确认的方法5.3 单个确认发布5.4 批量确认发布5.5 异步确认5.5.1 处理…...

通讯协议学习之路:RS422协议理论

通讯协议之路主要分为两部分&#xff0c;第一部分从理论上面讲解各类协议的通讯原理以及通讯格式&#xff0c;第二部分从具体运用上讲解各类通讯协议的具体应用方法。 后续文章会同时发表在个人博客(jason1016.club)、CSDN&#xff1b;视频会发布在bilibili(UID:399951374) 一、…...

剪映failed to initialize,cuda.is_available()为false解决

debug记录帖 错误1&#xff1a;打开剪映发现弹窗提示failed to initialize graphics backed for D3D11 错误2&#xff1a;torch版本、cuda版本&#xff08;之前的正常环境&#xff09;都对但是torch.cuda.is_available()为false 怀疑是显卡驱动的问题 打开Nvidia Geforce Exp…...

基于Spring Boot的LDAP开发全教程

写在前面 协议概述 LDAP&#xff08;轻量级目录访问协议&#xff0c;Lightweight Directory Access Protocol)是一种用于访问和维护分布式目录服务的开放标准协议,是一种基于TCP/IP协议的客户端-服务器协议&#xff0c;用于访问和管理分布式目录服务&#xff0c;如企业内部的…...

在 Linux 上保护 SSH 服务器连接的 8 种方法

SSH 是一种广泛使用的协议&#xff0c;用于安全地访问 Linux 服务器。大多数用户使用默认设置的 SSH 连接来连接到远程服务器。但是&#xff0c;不安全的默认配置也会带来各种安全风险。 具有开放 SSH 访问权限的服务器的 root 帐户可能存在风险。尤其是如果使用的是公共 IP 地…...

摩尔信使MThings的协议转换(数据网关)功能

摩尔信使MThings可以作为现场总线&#xff08;RS485&#xff09;和以太网的数据中枢&#xff0c;并拥有强大的Modbus协议转换功能。 数据网关功能提供协议转换和数据汇聚功能&#xff0c;可实现多维度映射&#xff0c;包括&#xff1a;不同的通道(总线)类型、协议类型&#xff…...

Mac安装Kali保姆级教程

Mac安装Kali保姆级教程 其他安装教程&#xff1a;使用VMware安装系统Window、Linux&#xff08;kali&#xff09;、Mac操作系统 1 虚拟机安装VM Fusion 去官网下载VM Fusion 地址&#xff1a;https://customerconnect.vmware.com/en/evalcenter?pfusion-player-personal-13 …...

利用Spring Boot框架做事件发布和监听

一、编写事件 1.编写事件类并集成spring boot 事件接口&#xff0c;提供访问事件参数属性 public class PeriodicityRuleChangeEvent extends ApplicationEvent {private final JwpDeployWorkOrderRuleDTO jwpDeployWorkOrderRuleDTO;public PeriodicityRuleChangeEvent(Obje…...

KingBase库模式表空间和客户端认证(kylin)

库、模式、表空间 数据库 数据库基集簇与数据库实例 KES集簇是由单个KES实例管理的数据库的集合KES集簇中的库使用相同的全局配置文件和监听端口、共享相关的进程和内存结构同一数据库集簇中的进程、相关的内存结构统称为实例 数据库 数据库是一个长期存储在计算机内的、有…...

h5的扫一扫功能 (非微信浏览器环境下)

必须在 https 域名下才生效 <template><div><van-field label"服务商编码" right-icon"scan" placeholder"扫描二维码获取" click-right-icon"getCameras" /> <div class"scan" :style"{disp…...

Typora 导出PDF 报错 failed to export as pdf. undefined 解决方案

情况 我想把一个很大的markdown 导出为 248页的pdf 然后就报错 failed to export as pdf. undefined 原因 &#xff1a; 个人感觉应该是图片太大了 格式问题之类导致的 解决 文件 -> 偏好设置 - > 导出 -> pdf -> 自定义 -> 把大小全部改为24mm (虽然图中是32 …...

[架构之路-239]:目标系统 - 纵向分层 - 中间件middleware

目录 前言&#xff1a; 一、中间件概述 1.1 中间件在软件层次中的位置 1.2 什么是中间件 1.3 为什么需要中间件 1.4 中间件应用场合&#xff08;应用程序不用的底层需求&#xff1a;计算、存储、通信&#xff09; 1.5 中间件分类 - 按内容分 二、嵌入式系统的中间件 2…...

javascript利用xhr对象实现http流的comet轮循,主要是利用readyState等于3的特点

//此文件 为前端获取http流 <!DOCTYPE html> <html xmlns"http://www.w3.org/1999/xhtml" lang"UTF-8"></html> <html><head><meta http-equiv"Content-Type" content"text/html; charsetUTF-8"/&g…...

【Mybatis源码】XPathParser解析器

XPathParser是Mybatis中定义的进行解析XML文件的类,此类用于读取XML文件中的节点文本与属性;本篇我们主要介绍XPathParser解析XML的原理。 一、XPathParser构造方法 这里我们介绍主要的构造方法 public XPathParser(InputStream inputStream, boolean validation, Propert…...

辉视智慧酒店解决方案助力传统酒店通过智能升级焕发新生

辉视智慧酒店解决方案基于强大的物联网平台&#xff0c;将酒店客控、网络覆盖、客房智能化控制、酒店服务交互等完美融合&#xff0c;打造出全方位的酒店智慧化产品。利用最新的信息化技术&#xff0c;我们推动酒店智慧化转型&#xff0c;综合运用前沿的信息科学和技术、消费方…...

文件和命令的查找与处理

1.命令查找 which which 接命令 2.文件查找 find 按文件名字查找 准确查找 find / -name "hosts" 粗略查找 find / -name "ho*ts" 扩展名查找 find / -name "*.txt" 按文件类型查找 find / -type f 文件查找 find / -ty…...

第七章:最新版零基础学习 PYTHON 教程—Python 列表(第三节 -Python程序访问列表中的索引和值)

有多种方法可以访问列表的元素,但有时我们可能需要访问元素及其所在的索引。让我们看看访问列表中的索引和值的所有不同方法。 目录 使用Naive 方法访问列表中的索引和值 使用列表理解访问列表中的索引和值...

利用ngx_stream_return_module构建简易 TCP/UDP 响应网关

一、模块概述 ngx_stream_return_module 提供了一个极简的指令&#xff1a; return <value>;在收到客户端连接后&#xff0c;立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量&#xff08;如 $time_iso8601、$remote_addr 等&#xff09;&a…...

R语言AI模型部署方案:精准离线运行详解

R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...

UDP(Echoserver)

网络命令 Ping 命令 检测网络是否连通 使用方法: ping -c 次数 网址ping -c 3 www.baidu.comnetstat 命令 netstat 是一个用来查看网络状态的重要工具. 语法&#xff1a;netstat [选项] 功能&#xff1a;查看网络状态 常用选项&#xff1a; n 拒绝显示别名&#…...

跨链模式:多链互操作架构与性能扩展方案

跨链模式&#xff1a;多链互操作架构与性能扩展方案 ——构建下一代区块链互联网的技术基石 一、跨链架构的核心范式演进 1. 分层协议栈&#xff1a;模块化解耦设计 现代跨链系统采用分层协议栈实现灵活扩展&#xff08;H2Cross架构&#xff09;&#xff1a; 适配层&#xf…...

[10-3]软件I2C读写MPU6050 江协科技学习笔记(16个知识点)

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16...

Typeerror: cannot read properties of undefined (reading ‘XXX‘)

最近需要在离线机器上运行软件&#xff0c;所以得把软件用docker打包起来&#xff0c;大部分功能都没问题&#xff0c;出了一个奇怪的事情。同样的代码&#xff0c;在本机上用vscode可以运行起来&#xff0c;但是打包之后在docker里出现了问题。使用的是dialog组件&#xff0c;…...

视频行为标注工具BehaviLabel(源码+使用介绍+Windows.Exe版本)

前言&#xff1a; 最近在做行为检测相关的模型&#xff0c;用的是时空图卷积网络&#xff08;STGCN&#xff09;&#xff0c;但原有kinetic-400数据集数据质量较低&#xff0c;需要进行细粒度的标注&#xff0c;同时粗略搜了下已有开源工具基本都集中于图像分割这块&#xff0c…...

宇树科技,改名了!

提到国内具身智能和机器人领域的代表企业&#xff0c;那宇树科技&#xff08;Unitree&#xff09;必须名列其榜。 最近&#xff0c;宇树科技的一项新变动消息在业界引发了不少关注和讨论&#xff0c;即&#xff1a; 宇树向其合作伙伴发布了一封公司名称变更函称&#xff0c;因…...

Git 3天2K星标:Datawhale 的 Happy-LLM 项目介绍(附教程)

引言 在人工智能飞速发展的今天&#xff0c;大语言模型&#xff08;Large Language Models, LLMs&#xff09;已成为技术领域的焦点。从智能写作到代码生成&#xff0c;LLM 的应用场景不断扩展&#xff0c;深刻改变了我们的工作和生活方式。然而&#xff0c;理解这些模型的内部…...

(一)单例模式

一、前言 单例模式属于六大创建型模式,即在软件设计过程中,主要关注创建对象的结果,并不关心创建对象的过程及细节。创建型设计模式将类对象的实例化过程进行抽象化接口设计,从而隐藏了类对象的实例是如何被创建的,封装了软件系统使用的具体对象类型。 六大创建型模式包括…...