【设计模式】【创建型模式】【02工厂模式】

系列文章

可跳转到下面链接查看下表所有内容https://blog.csdn.net/handsomethefirst/article/details/138226266?spm=1001.2014.3001.5501文章浏览阅读2次。系列文章大全https://blog.csdn.net/handsomethefirst/article/details/138226266?spm=1001.2014.3001.5501


目录

系列文章

1.简介

2.简单工厂模式

2.1 简介

2.1.1 组成结构

2.1.2 优点和缺点

2.1.3 应用场景

2.2 代码案例

2.2.1 主要步骤

2.2.2 代码

3.工厂方法模式

3.1简介

3.1.1 组成结构

3.1.2 优点和缺点

3.1.3 应用场景

3.2 代码案例

3.2.1 主要步骤

3.2.2 代码

4.抽象工厂模式

4.1简介

4.1.1 组成结构

4.1.2 优点和缺点

4.2 代码案例

4.2.1 主要步骤

4.2.2 代码

5.区别


1.简介

工厂模式主要是为创建对象提供过渡接口,以便将创建对象的具体过程屏蔽隔离起来,即定义一个用于创建对象的接口,让子类决定应该实例化哪个类,使类的实例化延迟到子类。 也可以这样理解,
工厂模式可以分为三类:
1.简单工厂模式(Simple Factory)
2.工厂方法模式(Factory Method)
3.抽象工厂模式(Abstract Factory)

2.简单工厂模式

2.1 简介

简单工厂模式是通过一个工厂类将所有对象的创建封装起来,通过传入不同的参数,工厂负责创建不同的具体对象。

建立对象的类是工厂,被建立的对象是产品,使用产品的人无需关心这个产品是怎样生产的,这样便降低了模块之间的耦合。

2.1.1 组成结构

1.工厂类:用来创建所有产品,通过传入的type不同,从而创建不同的产品。例如:本节案例代码中的Factory类。

2.抽象产品类:它一般是具体产品继承的父类或者实现的接口。例如:本节案例代码中的Car类。

3.具体产品类此类继承抽象产品类,自工厂类所创建的对象就是此角色的实例。例如:本节案例代码中的CarA类和CarB类。

特点:一个工厂生产所有的具体产品。

2.1.2 优点和缺点

优点:
1.将所有对象的创建集合到了一个工厂类中,客户端只需要调用工厂类的接口,传入不同的参数,而无需关注具体对象的创建过程。

2.可以通过工厂类创建不同的对象,而客户端无需改动,实现了客户端和具体所有对象的解耦,提高了代码的灵活性。
缺点:
1.每新增一个产品,则需要修改工厂类的逻辑,违反了开闭原则。
2.当产品足够多的时候,代码会过于臃肿,不利于维护。

2.1.3 应用场景

1.需要根据不同的参数类型创建不同的对象时。

2.2 代码案例

当前代码场景:

我们现在一个工厂要生产车型A和车型B

2.2.1 主要步骤

第一步:创建抽象产品类Car和创建抽象run函数
第二步:创建具体产品类车型A(CarA),实现其抽象run函数。
第三步:创建具体产品类车型B(CarB),实现其抽象run函数
第四步:创建工厂类Factory,并书写根据不同参数创建不同车型的createCar方法。
第五步:客户端(此例是main.cpp调用)创建工厂类对象,通过传递不同的参数,工厂创建不同的对象并返回给客户。

2.2.2 代码

// 抽象产品类
class Car
{public:virtual void run() = 0; // 抽象共同接口string cartype;
};// 具体产品类:车A
class CarA : public Car
{public:CarA(){cartype = "A 车";}virtual void run(){std::cout << "this is cartype = " << cartype << endl;}
};// 具体产品类:车B
class CarB : public Car
{public:CarB(){cartype = "B 车";}virtual void run(){std::cout << "this is cartype = " << cartype << endl;}
};// 工厂类
class Factory
{
public:Car *createCar(int type){switch (type){case CAR_TYPE_A:return new CarA();break;case CAR_TYPE_B:return new CarB();break;default:return nullptr;break;}}
};int main()
{Factory *mFactory = new Factory();           // c创建工厂类Car *acar = mFactory->createCar(CAR_TYPE_A); // 根据具体的参数创建对应的对象,此处是父类指针指向子类对象Car *bcar = mFactory->createCar(CAR_TYPE_B);acar->run();bcar->run();// 注意析构delete acar;delete bcar;delete mFactory;return 0;
}

3.工厂方法模式

3.1简介

工厂方法模式在简单工厂模式的基础上,去掉了简单工厂模式中的创建所有对象的方法,并提供了一个抽象生产产品的接口,并使其它可以被子类继承。这样在简单工厂模式里集中在工厂方法上的压力可以由工厂方法模式里不同的工厂子类来分担。
简单工厂模式的特点是一个工厂只生产所有产品,而工厂方法模式则是一个工厂只生产一个产品,这样当生产产品的时候,只需要确定是哪个工厂即可。

3.1.1 组成结构

1.抽象工厂类:是具体工厂类的父类,其包含一个具体工厂类必须实现的抽象接口。例如:本节案例代码的Factory类,抽象接口是createCar函数。
2.具体工厂类:对应其具体产品,用以创建对应的具体产品的对象。例如:本节案例代码的工厂A和工厂B,工厂A负责生产车型A, 工厂B负责生产车型B。
3.抽象产品类:它是具体产品继承的父类。例如:本节案例代码的Car类。
4.具体产品类:具体工厂类所创建的对象就是此类的实例。例如:本节案例代码的车型A类和车型B类。

3.1.2 优点和缺点

优点:
1.当需要增加新的工厂和产品的时候,可以不用更改现有代码,增加了代码的可扩展性,将对象的创建和使用进行了解耦。
缺点:
1.当产品种类过多时,由于每一种产品都需要实现一个工厂类,增加代码复杂性。

3.1.3 应用场景

当只有一类产品时。同时客户端需要生成不同的对象。

3.2 代码案例

此时场景为:
A工厂生产车型A,B工厂生产车型B

3.2.1 主要步骤

第一步:创建抽象产品类(Car类)。

第二步:创建抽象产品类的两个子类(CarA类和CarB类),其子类是具体产品类。

第三步:创建抽象工厂类(Factory类),并提供生产对象的抽象接口。

第四步:创建两个具体工厂类,一个是生产CarA的工厂类,一个是生产CarB的工厂类,并实现抽象方法,创建不同的产品对象。

第五步:客户端(此例是main.cpp调用)创建不同的工厂类对象,从而创建出多个不同的产品对象。

3.2.2 代码

// 抽象产品类
class Car
{public:virtual void run() = 0; // 抽象共同接口string cartype;
};// 具体产品类:车A
class CarA : public Car
{public:CarA(){cartype = "A 车";}virtual void run(){std::cout << "this is cartype = " << cartype << endl;}
};// 具体产品类:车B
class CarB : public Car
{public:CarB(){cartype = "B 车";}virtual void run(){std::cout << "this is cartype = " << cartype << endl;}
};// 抽象工厂类
class Factory
{
public:virtual Car *createCar() = 0;
};// 具体工厂类:工厂A(对应产品A)
class FactoryA : public Factory
{
public:virtual Car *createCar(){return new CarA();}
};
// 具体工厂类:工厂B(对应产品B)
class FactoryB : public Factory
{
public:virtual Car *createCar(){return new CarB();}
};int main()
{Factory *mFactoryA = new FactoryA(); // 创建具体工厂类ACar *acar = mFactoryA->createCar();  // 根据具体的工厂类创建对应的对象,此处是父类指针指向子类对象acar->run();Factory *mFactoryB = new FactoryB(); // 创建具体工厂类BCar *bcar = mFactoryB->createCar();  // 根据具体的工厂类创建对应的对象,此处是父类指针指向子类对象bcar->run();// 注意析构delete acar;delete bcar;delete mFactoryA;delete mFactoryB;return 0;
}

4.抽象工厂模式

之前的工厂方法模式一个工厂只生产一个产品,而在实际生活中,通过一个工厂会生产多种产品,比如:车的组成需要轮胎和方向盘,那么当出现多个产品的时候,便需要使用抽象工厂模式。

4.1简介

提供一个创建一系列相关或相互依赖的对象接口,而无需指定它们的具体类。其实抽象工厂模式就是多个工厂方法模式,只是因为工厂方法是一个工厂只创建一个产品,而抽象工厂模式是一个工厂创建多个产品。

4.1.1 组成结构

抽象工厂类:是具体工厂类的父类,其内部定义了创建多个产品对象的抽象接口,必须由具体工厂类实现。
例如:本节代码案例中的Factory类,抽象接口为createTire和createWheel。

具体工厂类:对应其具体产品,用以创建对应的具体产品的对象。
例如:本节代码案例中的 具体工厂(工厂白色)和具体工厂(工厂黑色)。

抽象产品类:定义了产品的共同接口,具体的产品类必须实现这个接口。工厂模式会由多个产品,因此抽象产品类也是多个。
例如:本节代码案例中的第一个抽象产品类(轮胎)和第二个抽象产品类(方向盘)。

具体产品类:是抽象工厂模式中具体创建的对象。
例如:本节代码案例中的白轮胎和黑轮胎,白方向盘和黑方向盘。

4.1.2 优点和缺点

优点:
创建产品族,将一系列的产品族,统一到一起进行创建
缺点:
扩展困难,当产品族中有新产品,比如新增发动机产品时,需要修改抽象工厂的接口。

4.2 代码案例

现在场景为:
白色工厂需要生产白色轮胎和白色方向盘。
黑色工厂需要生产黑色轮胎和黑色方向盘。
白色系列产品由白色工厂统一管理。
黑色系列产品由黑色工厂统一管理。

4.2.1 主要步骤

第一步:创建第一个抽象产品类(轮胎类)。

第二步:创建第一个抽象产品类的两个子类(白轮胎类和黑轮胎类),其子类是具体产品类。

第三步:创建第二个抽象产品类(方向盘类)。

第四步:创建第二个抽象产品类的两个子类(白方向盘类和黑方向盘类),其子类是具体产品类。

第五步:创建抽象工厂类(Factory类),并提供生产所有对象的两个抽象接口。

第六步:创建两个具体工厂类,一个是生产白色产品的白色工厂类,一个是生产黑色产品的白色工厂类,并实现两个抽象方法,创建不同的产品对象。

第七步:客户端(此例是main.cpp调用)创建不同的工厂类对象,从而创建出多个不同的产品对象。

4.2.2 代码

// 第一个抽象产品类:轮胎
class Tire
{public:virtual void whcihtype() = 0;string Tiretype;
};// 第一个抽象产类的第一个具体产品类:白轮胎
class WhiteTire : public Tire
{public:WhiteTire(){Tiretype = "白色轮胎";}virtual void whcihtype(){std::cout << "this is Tiretype = " << Tiretype << endl;}
};// 第一个抽象产类的第二个具体产品类:黑轮胎
class BlackTire : public Tire
{public:BlackTire(){Tiretype = "黑色轮胎";}virtual void whcihtype(){std::cout << "this is Tiretype = " << Tiretype << endl;}
};// 第二个抽象产品类:方向盘
class Wheel
{
public:virtual void whcihtype() = 0;string Wheeltype;
};// 第二个抽象产类的第一个具体产品类:白色方向盘
class WhiteWheel : public Wheel
{public:WhiteWheel(){Wheeltype = "白色方向盘";}virtual void whcihtype(){std::cout << "this is Wheeltype = " << Wheeltype << endl;}
};// 第二个抽象产类的第二个具体产品类:黑色方向盘
class BlackWheel : public Wheel
{public:BlackWheel(){Wheeltype = "黑色方向盘";}virtual void whcihtype(){std::cout << "this is Wheeltype = " << Wheeltype << endl;}
};// 抽象工厂类
class Factory
{
public:virtual Tire *createTire() = 0; // 抽象接口virtual Wheel *createWheel() = 0;
};// 具体工厂类:工厂白色(对应生产白色轮胎和白色方向盘)
class FactoryWhite : public Factory
{
public:virtual Tire *createTire(){return new WhiteTire(); // 生产白轮胎}virtual Wheel *createWheel(){return new WhiteWheel(); // 生产白方向盘}
};// 具体工厂类:工厂黑色(对应生产黑色轮胎和黑色方向盘)
class FactorBlock : public Factory
{
public:virtual Tire *createTire(){return new BlackTire(); // 生产黑轮胎}virtual Wheel *createWheel(){return new BlackWheel(); // 生产黑方向盘}
};int main()
{Factory *mFactoryWhite = new FactoryWhite();    // 创建具体工厂类:工厂白色Tire *mWhiteTire = mFactoryWhite->createTire(); // 创建白色轮胎mWhiteTire->whcihtype();Wheel *mWhiteWheel = mFactoryWhite->createWheel(); // 创建白色方向盘mWhiteWheel->whcihtype();Factory *mFactoryBlock = new FactorBlock();     // 创建具体工厂类:工厂黑色Tire *mBlockTire = mFactoryBlock->createTire(); // 创建黑色轮胎mBlockTire->whcihtype();Wheel *mBlockWheel = mFactoryBlock->createWheel(); // 创建黑色方向盘mBlockWheel->whcihtype();// 注意析构delete mFactoryWhite;delete mWhiteTire;delete mWhiteWheel;delete mFactoryBlock;delete mBlockTire;delete mBlockWheel;return 0;
}

5.区别

简单工厂模式:
一个抽象产品类,可以派生出多个具体产品类。
无抽象工厂类,只有一个工厂类,此工厂类负责生产所有产品。

工厂方法模式:
一个抽象产品类,可以派生出多个具体产品类。
一个抽象工厂类,可以派生出多个具体工厂类。
每个具体工厂类只能创建一个具体产品类的实例。

抽象工厂模式:
多个抽象产品类,每个抽象产品类可以派生出多个具体产品类。
一个抽象工厂类,可以派生出多个具体工厂类。
每个具体工厂类可以创建多个具体产品类的实例。

总结:

抽象产品类:工厂方法模式和简单工厂模式都只有一个抽象产品类,而抽象工厂模式有多个。

具体产品类:工厂方法模式和简单工厂模式都只能有一个抽象产品类派生出多个具体产品类,而抽象工厂模式可以从多个抽象产品类派生出多个具体产品类。

抽象工厂类:工厂方法模式和抽象工厂模式都有一个抽象工厂类,而简单工厂模式没有。

具体工厂类:简单工厂模式只有一个工厂类,而工厂方法模式和抽象工厂模式可以有多个具体工厂类。工厂方法模式的具体工厂类只能创建一个具体产品类的实例,而抽象工厂模式可以创建多个。

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

C++链接FTP服务器并下载数据(在qt中编写)

.pro文件 #------------------------------------------------- # # Project created by QtCreator 2024-07-16T13:19:03 # #-------------------------------------------------QT core gui networkgreaterThan(QT_MAJOR_VERSION, 4): QT widgetsTARGET untitled TE…

通过SchedulingConfigurer 接口完成动态定时任务

通过SchedulingConfigurer 接口完成动态定时任务 一.背景 在Spring中&#xff0c;除了使用Scheduled注解外&#xff0c;还可以通过实现SchedulingConfigurer接口来创建定时任务。它们之间的主要区别在于灵活性和动态性。Scheduled注解适用于固定周期的任务&#xff0c;一旦任…

【C++数据结构】二叉搜索树(超详细图解操作过程,超详细讲解代码实现)

目录 01.二叉搜索树的概念 02.二叉搜索树的操作过程 03.二叉搜索树的代码实现 &#xff08;1&#xff09;基本框架 &#xff08;2&#xff09;树的创建与销毁 &#xff08;3&#xff09;元素的查找 &#xff08;4&#xff09;元素的插入 &#xff08;5&#xff09;元素的…

Day71 代码随想录打卡|回溯算法篇---全排列

题目&#xff08;leecode T46&#xff09;&#xff1a; 给定一个不含重复数字的数组 nums &#xff0c;返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 方法&#xff1a;全排列是数学中的基础问题&#xff0c;也是回溯算法能解决的经典问题。全排列因为每个元素都会…

卷积神经网络学习问题总结

问题一&#xff1a; 深度学习中的损失函数和应用场景 回归任务&#xff1a; 均方误差函数&#xff08;MSE&#xff09;适用于回归任务&#xff0c;如预测房价、预测股票价格等。 import torch.nn as nn loss_fn nn.MSELoss() 分类任务&#xff1a; 交叉熵损失函数&…

AI算法19-偏最小二乘法回归算法Partial Least Squares Regression | PLS

偏最小二乘法回归算法简介 算法概述 偏最小二乘法模型可分为偏最小二乘回归模型和偏最小二乘路径模型。其中偏最小二乘回归模型是一种新型的多元统计方法&#xff0c;它集中了主成分分析、典型相关分析和线性回归的特点&#xff0c;特别在解决回归中的共线性问题具有无可比拟…

PX4 运行 make px4_sitl_default gazebo 报错

报错原因&#xff1a;最开始我把依赖一直都是在base环境下安装的&#xff0c;没有conda deactivate&#xff0c;而pip install的东西应该装在系统环境&#xff0c;不能装在base环境下&#xff0c;sudo apt 是装在系统环境的 1.检查ros 用鱼香ros安装 wget http://fishros.…

日活2.5亿的Twitter 使用了哪些数据库?

Twitter 使用什么数据库存储用户每天发送的数亿条推文&#xff1f;是 SQL、NoSQL 还是其它持久化存储系统&#xff1f; Twitter 使用什么数据库&#xff1f; 任何一个稍微有点规模的系统其存储层绝不会只使用一种数据库&#xff0c;服务于数以亿计用户的Twitter更是如此。Twit…

《YOLOv10改进实战专栏》专栏介绍 专栏目录

《YOLOv10改进实战专栏》介绍及目录 YOLOv10官方仓库地址 专栏地址&#xff1a;点击跳转 专栏导航如下&#xff1a; &#x1f380;基础入门篇&#x1f380; 万字长文&#xff0c;小白新手怎么开始做YOLO实验&#xff0c;从零开始教&#xff01;整体思路在这里&#xff0c;科研指…

Vue学习---vue cli 项目创建

使用的编辑工具webStorm 创建例子: hello vue create hello 选择 vue3 进行创建 运行 npm run serve 测试访问&#xff1a;http://localhost:8080 改动内容重新编译&#xff1a; npm run build dist 目录就是编译后的可运行内容

浅谈C嘎嘎类与对象

本篇文章与大家浅谈一下C嘎嘎的类与对象知识点 类的定义 关键字&#xff1a;class 语法格式&#xff1a; class 类名 { }&#xff1b;//这里的分号不能少 此外&#xff0c;class有三个属性分别是private、public、protected&#xff0c;这三个属性是干啥的&#xff0c;相…

MSPM0G3507——时钟主频拉到80MHZ

先点开使用时钟树 在配置时钟界面这样配置

Ghost Browser指纹浏览器年+IPXProxy代理IP组合:SheIn卖家必看

SheIn是一家时尚电商公司&#xff0c;其用户数量近年来增长迅速&#xff0c;在全球的知名度越来越高。SheIn跨境电商卖家想要提升店铺曝光和排名&#xff0c;从而增加销量和信誉的话&#xff0c;就需要满足独立IP、模拟设备参数、独立环境等条件。同时满足这些条件的话就需要用…

生成式人工智能(AI)的未来

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

windows qt编译报错 无法打开包括文件: “EGL/egl.h”: No such file or directory

windows mingw32 qt creator QtAV 推荐ffmpeg依赖包 QT5.14.2 如果出现&#xff1a;无法打开包括文件: “EGL/egl.h”: No such file or directory 可能是Qt6的问题.在QT5上安装。 编译步骤&#xff1a; git clone https://github.com/wang-bin/QtAV.git cd QtAV &&…

【深度学习教程】

文章目录 pytorch官方教程知识蒸馏&#xff1a;https://pytorch.org/tutorials/beginner/knowledge_distillation_tutorial.html 李宏毅-机器学习/深度学习https://speech.ee.ntu.edu.tw/~hylee/ml/2021-spring.phphttps://speech.ee.ntu.edu.tw/~hylee/ml/2022-spring.phphttp…

最新Qt6的下载与成功安装详细介绍

引言 Qt6 是一款强大的跨平台应用程序开发框架&#xff0c;支持多种编程语言&#xff0c;最常用的是C。Qt6带来了许多改进和新功能&#xff0c;包括对C17的支持、增强的QML和UI技术、新的图形架构&#xff0c;以及构建系统方面的革新。本文将指导你如何在Windows平台上下载和安…

第五章:卷-将磁盘挂载到容器

本章内容包括&#xff1a; 创建多容器pod创建一个可在容器间共享磁盘存储的卷在pod中使用git仓库将持久性存储挂载到pod使用预先配置的持久性存储动态调配持久存储 在前面说过&#xff0c;pod类似逻辑主机&#xff0c;在逻辑主机中运行的进程共享如CPU、RAM、网络接口等资源&am…

一分钟了解什么是1U,2U服务器?

一、什么是1U&#xff0c;2U服务器&#xff1f; 什么是1U服务器呢&#xff1f;所谓的1U服务器就是一种高可用高密度的低成本服务器平台&#xff0c;U是服务器机箱的高度 1U等于4.45厘米 &#xff0c;那3U就是3x4.5CM了。 u(unit的缩略语)是一种表示组合式机架外部尺寸的单位&a…

【时时三省】tessy 集成测试:小白入门指导手册

目录 1,创建集成测试模块且分析源文件 2,设置测试环境 3,TIE界面设置相关函数 4,SCE界面增加用例 5,编辑数据 6,用例所对应的测试函数序列 7,添加 work task 函数 8,为测试场景添加函数 9,为函数赋值 10,编辑时间序列的数值 11,执行用例 12,其他注意事项…