面试十 简单工厂、工厂方法、抽象工厂

/*简单工厂 Simple Factory:优点:把对象的创建封装在一个接口函数里面,通过传入不同的标识,返回创建的对象,客户不用自己负责new对象缺点 :提供创建对象实例的接口函数不闭合,不能对修改关闭,对扩展开放工厂方法 Factory Method优点:Factory基类,提供一个纯虚函数(创建产品),定义派生类(具体产品的工厂)负责创建对应的产品,可以做到不同的产品在不同的工厂创建,能够对现有工厂,以及产品的修改关闭缺点:实际上,很多产品是有关联关系的,属于一个产品族,不应该放在不同的工厂取创建,这样一不符合实际的产品对象创建逻辑,二是工厂类太多了,不好维护抽象工厂 Abstract Factory把有关联关系的,属于一个产品族的所有产品创建的接口函数,放在一个抽象工厂里面AbstractFactory,派生类(具体产品的工厂)应该负责创建该产品簇里面的所有产品工厂模式: 主要是封装了对象的创建*/

简单工厂模式(Simple Factory Pattern):

  • 简单工厂模式是一种由一个工厂类根据传入的参数决定创建哪一种产品类的实例的设计模式。
  • 客户端通过向工厂类传递指定的参数来获取所需的产品对象,而无需直接实例化具体的产品类。
  • 简单工厂模式的核心是一个工厂类,它包含了必要的逻辑判断,根据不同的条件创建不同的产品对象。
class Car
{
public:Car(string name) :_name(name) {}virtual void show() = 0;
protected:string _name;
};
class Bmw:public Car
{
public:Bmw(string name) :Car(name) {}void show(){cout << "获取了一辆宝马汽车:" << _name << endl;}
};class Audi :public Car
{
public:Audi(string name) :Car(name) {}void show(){cout << "获取了一辆奥迪汽车:" << _name << endl;}
};// 产品枚举
enum Cartype
{BMW, AUDI
};
// 简单工厂类
class SimpleFactory
{
public:// 用户想要创建一个对象,只需要知道名称就可以了Car* createCar(Cartype ct){switch (ct){case BMW:return new Bmw("x6");case AUDI:return new Audi("a8");default:cerr << "传入参数错误:" << ct << endl;}return nullptr;}
};int main()
{unique_ptr<SimpleFactory> fac(new SimpleFactory());unique_ptr<Car> p1(fac->createCar(BMW));unique_ptr<Car> p2(fac->createCar(AUDI));p1->show();p2->show();return 0;
}

工厂方法模式(Factory Method Pattern):

  • 工厂方法模式定义了一个用于创建对象的接口,但将具体创建工作推迟到子类中。
  • 工厂方法模式通过在父类中定义一个创建对象的抽象方法,让子类来实现这个方法来创建具体的产品对象。
  • 每个具体的工厂类都负责创建一类产品。

// 系列产品
class Car
{
public:Car(string name) :_name(name) {}virtual void show() = 0;
protected:string _name;
};
class Bmw:public Car
{
public:Bmw(string name) :Car(name) {}void show(){cout << "获取了一辆宝马汽车:" << _name << endl;}
};
class Audi :public Car
{
public:Audi(string name) :Car(name) {}void show(){cout << "获取了一辆奥迪汽车:" << _name << endl;}
};// 基类(包含纯虚函数,不能实例化对象)
class Factory
{
public:virtual Car* createCar(string name) = 0;
};
// 宝马汽车工厂,负责生产宝马汽车
class BmwFac: public Factory
{
public:Car* createCar(string name){return new Bmw(name);}
};// 奥迪汽车工厂,负责生产奥迪汽车
class AudiFac :public Factory
{
public:Car* createCar(string name){return new Audi(name);}
};int main()
{unique_ptr<Factory> bmwfty(new BmwFac());unique_ptr<Factory> audifty(new AudiFac());unique_ptr<Car> p1 (bmwfty->createCar("X6"));unique_ptr<Car> p2 (audifty->createCar("A8"));p1->show();p2->show();return 0;
}

抽象工厂模式(Abstract Factory Pattern):

  • 抽象工厂模式提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。
  • 抽象工厂模式包含一个抽象工厂类,它声明了一组用于创建一系列产品的抽象方法。每个具体的工厂类负责创建一族产品,通常对应于一个产品等级结构。
  • 抽象工厂模式关注的是一系列相关的产品对象的创建,而不是单个产品对象的创建。
// 系列产品1:汽车
class Car
{
public:Car(string name) :_name(name) {}virtual void show() = 0;
protected:string _name;
};
class Bmw:public Car
{
public:Bmw(string name) :Car(name) {}void show(){cout << "获取了一辆宝马汽车:" << _name << endl;}
};
class Audi :public Car
{
public:Audi(string name) :Car(name) {}void show(){cout << "获取了一辆奥迪汽车:" << _name << endl;}
};
// 系列产品2:车灯
class Light
{
public:virtual void show() = 0;
};
class BmwLight : public Light
{
public:void show() { cout << "BMW light!" << endl; }
};
class AudiLight : public Light
{
public:void show() { cout << "Audi light!" << endl; }
};// 工厂方法 => 抽象工厂(对有一组关联关系的产品簇提供产品对象的统一创建)
class AbstractFactory
{
public:virtual Car* createCar(string name) = 0; // 工厂方法 创建汽车virtual Light* createCarLight() = 0; // 工厂方法 创建汽车关联的产品,车灯
};
// 宝马工厂
class BMWFactory : public AbstractFactory
{
public:Car* createCar(string name){return new Bmw(name);}Light* createCarLight(){return new BmwLight();}
};
// 奥迪工厂
class AudiFactory : public AbstractFactory
{
public:Car* createCar(string name){return new Audi(name);}Light* createCarLight(){return new AudiLight();}
};int main()
{unique_ptr<AbstractFactory> bmwfty(new BMWFactory());unique_ptr<AbstractFactory> audifty(new AudiFactory());unique_ptr<Car> p1(bmwfty->createCar("X6"));unique_ptr<Car> p2(audifty->createCar("A8"));unique_ptr<Light> l1(bmwfty->createCarLight());unique_ptr<Light> l2(audifty->createCarLight());p1->show();l1->show();p2->show();l2->show();return 0;
}

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

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

相关文章

Git一点通

1.Git的优势 Git是一个伟大的版本管理工具&#xff0c;比之svn&#xff0c;具有以下优势&#xff1a; 分布式版本控制&#xff1a;Git是一种分布式版本控制系统&#xff0c;每个开发者都拥有自己的完整代码库&#xff0c;不需要依赖网络连接就可以进行版本控制、合并和提交操作…

了解测试用例与测试场景

测试用例和测试场景是综合测试中最常见的两种测试工件。正确获得这两个可交付成果对于产品成功至关重要&#xff0c;因为它可以让软件开发团队和测试人员更高效地工作。然而&#xff0c;在 QA 测试中&#xff0c;测试场景和测试用例之间的差异可能会在转换过程中丢失。 测试用例…

opengl日记10-opengl使用多个纹理示例

文章目录 环境代码CMakeLists.txt文件内容不变。fragmentShaderSource.fsvertexShaderSource.vsmain.cpp 总结 环境 系统&#xff1a;ubuntu20.04opengl版本&#xff1a;4.6glfw版本&#xff1a;3.3glad版本&#xff1a;4.6cmake版本&#xff1a;3.16.3gcc版本&#xff1a;10.…

66、将同图片下的多个不同类别的xml标注文件合并成一个xml标注文件-labelImg格式

基本思想:手中有一套抽烟的数据集是labelimg格式,但是没有人物标注的数据集,因此使用自动化标注脚本将图片过滤一边,进行生成labelimg文件,只含有80类别的人物标注xml,然后使用脚本将生成标注的人物xml和手中有的抽烟xml进行合并,生成一份xml文件 代码 # -*- coding: u…

Linux查看8080端口是否启用

在Linux系统中&#xff0c;您可以使用几种不同的命令来检查8080端口是否被启用或正在被某个进程使用。以下是几种常用的方法&#xff1a; 使用lsof命令&#xff1a; sudo lsof -i :8080如果8080端口被某个进程使用&#xff0c;lsof命令将列出相关信息。如果没有输出&#xff0c…

docker和kubectl客户端安装Linux

一、docker安装 1.配置yum源&#xff08;系统组&#xff09; 2.查看可安装docker的所有版本 yum provides docker3.安装最新版本dockers yum install docker3.1确定版本没问题输入 y 4.验证 docker -v5.开启私有仓库的证书验证&#xff0c;没有创建一个daemon.json sudo vi…

设计模式之工厂方法模式解析

工厂方法模式 1&#xff09;问题 简单工厂模式 当需要引入新产品时&#xff0c;由于静态工厂方法通过所传入参数的不同来创建不同的产品&#xff0c;需要修改工厂类的源代码。 2&#xff09;概述 针对不同的产品提供不同的工厂&#xff0c;系统提供一个与产品等级结构对应…

我的保研材料全部损坏了!这个压缩包文件格式未知或数据已经被损坏不可预料的压缩文件末端

求助各位友友&#xff0c;我的保研材料全部没了&#xff01; 之前为了清理D盘&#xff0c;把之前保研期间准备的几个G的材料全部压缩放在了U盘&#xff0c;但是现在却损坏打不开了&#xff0c;之前为了省事也没有添加过“恢复记录”&#xff01;&#xff01;&#xff01; 先声…

阿赵UE学习笔记——20、角色蓝图和动画蓝图

阿赵UE学习笔记目录 大家好&#xff0c;我是阿赵。   继续学习虚幻引擎的使用。这次来看看角色控制动画相关的东西&#xff0c;主要用到了动画蓝图和角色蓝图。 一、动画蓝图 之前分析过&#xff0c;蓝图对于虚幻引擎来说&#xff0c;是存在于各个系统里面的&#xff0c;相当…

js 替换数组中的部分文字内容

用js 把[ "2024-03-20实时", "2024-03-20日前", "运行日实时", "运行日日前"]中把所有的“运行日”替换成 “2023” 可以使用 JavaScript 的 Array.prototype.map() 方法来遍历数组&#xff0c;并使用 String.prototype.replace() 方…

【0274】从shared init file或local init file加载relation cache(2 - 1)

上一篇: 【0273】深入分析 relcache(relation descriptor cache)初始化第一阶段(1) 【0264】深入分析relcache(relation descriptor cache)缓存初始化第2阶段(2) 1. 前言 本文内容是作为《【0264】深入分析relcache(relation descriptor cache)缓存初始化第2阶段…

智慧公厕:卫生、便捷、安全的新时代厕所变革

在城市快速发展的背景下&#xff0c;公共厕所的建设和管理变得越来越重要。智慧公厕作为厕所变革的一项全新举措&#xff0c;通过建立公共厕所全面感知监测系统&#xff0c;以物联网、互联网、大数据、云计算、自动化控制技术为支撑&#xff0c;实现对公共厕所的智能化管理和运…

FPGA学习_时序约束以及VIVADO时序报告

文章目录 前言时序约束的目的一、时序约束种类1、约束主时钟2、约束衍生时钟3、约束虚拟时钟4、input delay5、output delay6、约束异步时钟组7、约束互斥时钟8、假路径约束9、多周期约束 二、VIVADO时序报告三、从时序的角度看为什么寄存器赋值慢一拍 前言 一边学习一边补充当…

消除 Git diff 中的换行符差异(Linux)

通常编辑器默认使用的换行符是跟随操作系统的&#xff0c;而windows操作系统上修改的代码&#xff0c;其换行符会被转成win的\r\n,在提交代码时会显示大量改动&#xff08;对于sh脚本还会存在无法执行的问题&#xff09;&#xff0c;这时候我们可以通过设置git自动转成unix格式…

容器中的大模型(三)| 利用大语言模型:容器化高效地部署 PDF 解析器实践...

作者&#xff1a;宋文欣&#xff0c;智领云科技联合创始人兼CTO 01 简介 大语言模型&#xff08;LLMs&#xff09;正逐渐成为人工智能领域的一颗璀璨明星&#xff0c;它们的强大之处在于能够理解和生成自然语言&#xff0c;为各种应用提供了无限可能。为了让这些模型更好地服务…

【Hadoop】Hadoop 编译源码

目录 为什么要源码编译Hadoop 编译源码1前期工作准备2jar 包安装2.1安装 Maven2.2安装 ant2.3安装 glibc-headers 和 g2.4安装 make 和 cmake2.5安装 protobuf2.6安装 openssl 库2.7安装 ncurses-devel 库 3编译源码3.1解压源码到 /opt/ 目录3.2 进入到 hadoop 源码主目录 /opt…

原生小程序开发的父子组件传值,兄弟组件传值

1.父子传值&#xff0c;父组件通过属性的方式去给子组件传递值&#xff0c;子组件在properties属性去接收父组件传递过来的值&#xff1a; 父组件部分&#xff1a; <view class"pcolor"><customer id"child" bind:changSex"changSex"…

AI时代Python金融大数据分析实战:ChatGPT让金融大数据分析插上翅膀

❤️作者主页&#xff1a;小虚竹 ❤️作者简介&#xff1a;大家好,我是小虚竹。2022年度博客之星评选TOP 10&#x1f3c6;&#xff0c;Java领域优质创作者&#x1f3c6;&#xff0c;CSDN博客专家&#x1f3c6;&#xff0c;华为云享专家&#x1f3c6;&#xff0c;掘金年度人气作…

UniTask 异步任务

文章目录 前言一、UniTask是什么&#xff1f;二、使用步骤三、常用的UniTask API和示例1.编写异步方法2.处理异常3.延迟执行4.等待多个UniTask或者一个UniTas完成5.异步加载资源示例6.手动控制UniTask的完成状态7.UniTask.Lazy延迟任务的创建8.后台线程切换Unity主线程9.不要返…

第二证券|热度飙升,出境游人数有望破亿,这些概念股被机构盯上

在免签和航班批量康复的方针利好下&#xff0c;本年出境游商场迎来炽热升温。 清明出境游有望爆火 3月20日&#xff0c;Airbnb爱彼迎在北京举行春季出境游趋势发布会举行。 爱彼迎中国数据显示&#xff0c;本年清明节期间的出境游查找热度已经超出2023年同期的2.5倍&#xf…