类的设计模式——单例、工厂以及建造者模式

news2025/5/29 6:46:44

1.单例模式

1.1 饿汉模式

单例模式:一个类只能创建一个对象,这个设计模式可以保证系统中该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。

饿汉模式指在程序初始化时就创建一个唯一的实例对象。适用于多线程环境。他的初始化发生在Singleton Singleton::sig这里。

#include <iostream>
#include <memory>

class Singleton{
    static Singleton sig;
private:
    Singleton():_data(100){std::cout<<"我被初始化了"<<std::endl;}
    ~Singleton(){}
public:
    Singleton(const Singleton& sig) = delete;
    Singleton operator=(const Singleton& sig) = delete;
    Singleton& getSig()
    {
        return sig;
    }
private:
    int _data;
};

Singleton Singleton::sig;


int main()
{
    return 0;
}

1.2 懒汉模式

懒汉模式第一次使用单例对象的时候创建实例对象。

这里右两份代码,一份是通过锁来完成一份是通过C++11的static特性完成

class Singleton2
{
    static Singleton2 *sig2;
    static pthread_mutex_t mutex;

private:
    Singleton2() : _data(100)
    {
        std::cout << "我被初始化了 2" << std::endl;
        pthread_mutex_init(&mutex, nullptr);
    }
    ~Singleton2() { pthread_mutex_destroy(&mutex); }

public:
    Singleton2(const Singleton2 &sig) = delete;
    Singleton2 operator=(const Singleton2 &sig) = delete;
    static Singleton2* getSig()
    {
        if (sig2 == nullptr)
        {
            pthread_mutex_lock(&mutex);
            if (sig2 == nullptr)
                sig2 = new Singleton2;
            pthread_mutex_unlock(&mutex);
        }
        return sig2;
    }

private:
    int _data;
};

Singleton2 *Singleton2::sig2 = nullptr;
pthread_mutex_t Singleton2::mutex;
template <typename T>
class Singleton
{
private:
    Singleton() {}
    ~Singleton() {}

public:
    Singleton(const Singleton &) = delete;
    Singleton &operator=(const Singleton &) = delete;
    static T &getInstance()
    {
        static Singleton _eton;
        return _eton;
    }
};

2.工厂模式

工厂模式是一种创建型设计模式,他提供一种创建对象的最佳方式。在工厂模式中,我们创建对象时不会对上层暴露创建逻辑(方便用户操作),而是通过使用一个共同结构来指向新创建的对象,以此实现创建 使用的分离

2.1 简单工厂模式

简单工厂实现由一个工厂对象通过类型决定创建出来指定产品类的实例。

class product_abstract_class{
public:
    product_abstract_class(){}
    virtual void show() = 0;
};

class product1_class : public product_abstract_class
{
public:
    product1_class(){}
    void show() override
    {std::cout<<"this is product 1"<<std::endl;}
};

class product2_class : public product_abstract_class
{
public:
    product2_class(){}
    void show() override
    {std::cout<<"this is product 2"<<std::endl;}
};

class factory{
public:
    static std::shared_ptr<product_abstract_class> generate_instances(const std::string &name)
    {
        if(name == "product 1") return std::make_shared<product1_class>();
        else return std::make_shared<product2_class>();
    }
};

2.2 工厂方法模式

工厂方法模式:在简单工厂模式下新增多个工厂,每个产品对应一个工厂。

class product_abstract_class{
public:
    product_abstract_class(){}
    virtual void show() = 0;
};

class product1_class : public product_abstract_class
{
public:
    product1_class(){}
    void show() override
    {std::cout<<"this is product 1"<<std::endl;}
};

class product2_class : public product_abstract_class
{
public:
    product2_class(){}
    void show() override
    {std::cout<<"this is product 2"<<std::endl;}
};

// class factory{
// public:
//     static std::shared_ptr<product_abstract_class> generate_instances(const std::string &name)
//     {
//         if(name == "product 1") return std::make_shared<product1_class>();
//         else return std::make_shared<product2_class>();
//     }
// };

class factory_abstract_class{
public:
    virtual std::shared_ptr<product_abstract_class> create() = 0;    
};

class product1_factory : public factory_abstract_class
{
public:
    std::shared_ptr<product_abstract_class> create() override
    {
        return std::make_shared<product1_class>();
    }
};

class product2_factory : public factory_abstract_class
{
public:
    std::shared_ptr<product_abstract_class> create() override
    {
        return std::make_shared<product2_class>();
    }
};

工厂方法模式每次增加一个产品时,都需要增加一个具体的产品类和工厂类,这会让系统中类的个数成倍增加,在一定程度上增加了系统的耦合度。

而且工厂方法模式会让代码感觉很冗余,但是又必不可少,无法优化

2.3 抽象工厂模式

抽象工厂模式在工厂方法模式中引入了工厂等级结构。

我感觉就是工厂分类,比如香蕉苹果属于水果那么就通过水果工厂生产,狗和猫属于动物就通过动物工厂来生产

class abs_fruit
{
public:
    abs_fruit(){}
    virtual void show() = 0;
};

class apple_fruit : public abs_fruit
{
public:
    void show() override
    {std::cout<<"this is apple"<<std::endl;}
};

class banana_fruit : public abs_fruit
{
public:
    void show() override
    {std::cout<<"this is banana"<<std::endl;}
};


class abs_animal
{
public:
    abs_animal(){}
    virtual void show() = 0;
};

class cat : public abs_animal
{
public:
    void show() override
    {std::cout<<"this is cat"<<std::endl;}
};

class dog : public abs_animal
{
public:
    void show() override
    {std::cout<<"this is dog"<<std::endl;}
};

class abs_factory
{
public:
    virtual std::shared_ptr<abs_fruit> create_fruit(const std::string &name) = 0;
    virtual std::shared_ptr<abs_animal> create_animal(const std::string &name) = 0;
};

class fruit_factory : public abs_factory
{
public:
    std::shared_ptr<abs_fruit> create_fruit(const std::string &name) override
    {
        if(name == "apple") return std::make_shared<apple_fruit>();
        else if(name == "banana") return std::make_shared<banana_fruit>();
        else return std::shared_ptr<abs_fruit>();
    }

    virtual std::shared_ptr<abs_animal> create_animal(const std::string &name) override
    {
        return std::shared_ptr<abs_animal>();
    }
};

class animal_factory : public abs_factory
{
public:
    std::shared_ptr<abs_fruit> create_fruit(const std::string &name) override
    {
        return std::shared_ptr<abs_fruit>();
    }

    virtual std::shared_ptr<abs_animal> create_animal(const std::string &name) override
    {
        if(name == "cat") return std::make_shared<cat>();
        else if(name == "dog") return std::make_shared<dog>();
        else return std::shared_ptr<abs_animal>();
    }
};

class produce_factory
{
public:
    static std::shared_ptr<abs_factory> create_factory(const std::string& name)
    {
        if(name == "fruit") return std::make_shared<fruit_factory>();
        else if(name == "animal") return std::make_shared<animal_factory>();
        else return std::shared_ptr<abs_factory>();
    }
};

3.建造者模式

建造者模式是一种创建型设计模式,使用多个简单的对象一步一步构建一个复杂对象,能够将一个复杂对象的构建与它的表示分离,提供一种创建对象的最佳方式。

主要用于解决对象的构建过于复杂的问题

建造者模式的五个核心类

抽象产品类:用于给具体的产品类继承

具体的产品类:具体的产品对象类

抽象Builder(建造)类:创建一个产品对象所需的各个部件的抽象接口

具体产品的Builder类:实现抽象接口,构建各个部件

指挥者Director类:统一组建过程,提供给调用者使用,通过指挥者来构造产品

//---------------------------------------建造者模式-----------------------------------------------

//抽象产品类
class Computer
{
public:
    using ptr = std::shared_ptr<Computer>; //为std::shared_ptr<Computer>创建类型别名ptr    using等价于typedef
    Computer(){}
    void setboard(const std::string &board){_board = board;}
    void setdisplay(const std::string &display){_display = display;}
    virtual void setos() = 0;
    virtual ~Computer(){};
public:
    std::string to_string()
    {
        std::string retstr = "os : "+_os+" "+"board : "+_board+" "+"display : "+_display;
        return retstr;
    }
protected:
    std::string _board;
    std::string _display;
    std::string _os;
};

//具体产品类
class MacBook : public Computer
{
public:
    using ptr = std::shared_ptr<MacBook>; 
    void setos() override
    { _os = "Max Os X12";}
};

//抽象建造者类
class Builder
{
public:
    using ptr = std::shared_ptr<Builder>;
    virtual void builder_board(const std::string &board) = 0;
    virtual void builder_display(const std::string &display) = 0;
    virtual void builder_os() = 0;
public:
    virtual Computer::ptr builder() = 0; // 用于建造对象
};

//具体产品建造者类
class MacBook_Builder : public Builder
{
public:
    MacBook_Builder() :_computer(new MacBook()){}
public:
    void builder_board(const std::string &board)
    {
        _computer->setboard(board);
    }
    void builder_display(const std::string &display)
    {
        _computer->setdisplay(display);
    }
    void builder_os()
    {
        _computer->setos();
    }
    Computer::ptr builder()
    {
        return _computer;
    }
private:
    Computer::ptr _computer;
};

//指挥者类
class Director
{
public:
    Director(Builder* builder) :_builder(builder){}
    void construct(const std::string &board,const std::string &display)
    {
        _builder->builder_board(board);
        _builder->builder_display(display);
        _builder->builder_os();
    }
private:
    Builder::ptr _builder;
};


int main()
{
    Builder* dp = new MacBook_Builder;
    std::shared_ptr<Director> dir(new Director(dp)); 
    dir->construct("英特尔主板","VOC显示器");
    Computer::ptr tmp = dp->builder();    //这才是我们的目标对象(想要创建的对象)
    std::cout<<tmp->to_string()<<std::endl;

    return 0;
}

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

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

相关文章

STM32之看门狗(IWDG)

一、看门狗外设的原理与应用 背景说明 随着单片机的发展&#xff0c;单片机在家用电器、工业自动化、生产过程控制、智能仪器仪表等领域的应用越来越广泛。然而处于同一电力系统中的各种电气设备通过电或磁的联系彼此紧密相连&#xff0c;相互影响&#xff0c;由于运行方式的…

跟着华为去变革 ——读《常变与长青》有感

《常变与长青》&#xff0c;是华为郭平总2024年上市的著作。走进这本书&#xff0c;我们能够清晰看到华为30多年的成长过程和伴随期间的变革历程&#xff1a;从一家设备代理商开始&#xff0c;起步蹒跚&#xff0c;砥砺前行&#xff0c;在闯过一个又一个磨难之后&#xff0c;成…

图像分割技术的实现与比较分析

引言 图像分割是计算机视觉领域中的一项基础技术&#xff0c;其目标是将数字图像划分为多个图像子区域&#xff08;像素的集合&#xff09;&#xff0c;以简化图像表示&#xff0c;便于后续分析和理解。在医学影像、遥感图像分析、自动驾驶、工业检测等众多领域&#xff0c;图…

node.js配置变量

一、下载安装包 1、官网下载 大家可以在官网下载&#xff0c;适合自己电脑以及项目的需要的版本。 二、node.js安装 1、安装 双击下载的安装包文件&#xff0c;通常为 .exe 或 .msi 格式&#xff08;Windows&#xff09;或 .dmg 格式&#xff08;Mac&#xff09;。系统会…

Ubuntu+Docker+内网穿透:保姆级教程实现安卓开发环境远程部署

文章目录 前言1. 虚拟化环境检查2. Android 模拟器部署3. Ubuntu安装Cpolar4. 配置公网地址5. 远程访问小结 6. 固定Cpolar公网地址7. 固定地址访问 前言 本文将详细介绍一种创新性的云开发架构&#xff1a;基于Ubuntu系统构建Android仿真容器环境&#xff0c;并集成安全隧道技…

计算机网络总结(物理层,链路层)

目录 第一章 概述 1.基本概念 2.- C/S模式&#xff0c;B/S模式&#xff0c;P2P模式 3.- LAN,WAN,MAN,PAN的划分 4.电路交换与分组交换&#xff0c;数据报交换和虚电路交换 第二章 物理层 1.信号编码&#xff1a;不归零编码&#xff0c;曼切斯特编码 2.几种复用技术的特…

TIGER - 一个轻量高效的语音分离模型,支持人声伴奏分离、音频说话人分离等 支持50系显卡 本地一键整合包下载

TIGER 是一种轻量级语音分离模型&#xff0c;通过频段分割、多尺度及全频帧建模有效提取关键声学特征。该项目由来自清华大学主导研发&#xff0c;通过频率带分割、多尺度以及全频率帧建模的方式&#xff0c;有效地提取关键声学特征&#xff0c;从而实现高效的语音分离。 TIGER…

无人机降落伞设计要点难点及原理!

一、设计要点 1. 伞体结构与折叠方式 伞体需采用轻量化且高强度的材料&#xff08;如抗撕裂尼龙或芳纶纤维&#xff09;&#xff0c;并通过多重折叠设计&#xff08;如三重折叠缝合&#xff09;减少展开时的阻力&#xff0c;同时增强局部承力区域的强度。 伞衣的几何参数&am…

20250526给荣品PRO-RK3566的Android13单独编译boot.img

./build.sh init ./build.sh -K ./build.sh kernel 20250526给荣品PRO-RK3566的Android13单独编译boot.img 2025/5/26 15:25 缘起&#xff1a;需要给荣品PRO-RK3566的Android13单独编译内核&#xff0c;但是不想编译整个系统。于是&#xff1a; 如果特调试某些特别的改动/文件…

构建版本没mac上传APP方法

在苹果开发者的app store connect上架Ios应用的时候&#xff0c;发现需要使用xode等软件来上传iOS的APP。 但是不管是xcode也好&#xff0c;transporter也好&#xff0c;还是命令行工具也好&#xff0c;都必须安装在mac电脑才能使用&#xff0c;。 假如没有mac电脑&#xff0…

如何解决大模型返回的JSON数据前后加上```的情况

环境说明 springboot 应用使用dashscope-sdk-java对接阿里百练 deepseek v3模型 问题表现 已经指定了输出json格式&#xff0c;但指令不明确&#xff0c;输出JSON格式的写法如下 注&#xff1a;提示词一开始是能正常功能的&#xff0c;但过了几天就出现了异常&#xff0c;原…

服务器异常数据问题解决 工具(tcpdump+wireshark+iptables)

问题&#xff1a; 某天一客户反馈&#xff0c;后台页面上显示的设备数据异常增长。现场实际只有2w台设备安装了助手(客户端)&#xff0c;但是后台显示有16w的助手设备&#xff0c;并且还在持续且快速的增长。这些数据会被加载到缓存&#xff0c;时间久了&#xff0c;服务端程序…

综合实现案例 LVS keepalived mysql 等

基于企业级高可用架构的 Linux 案例&#xff0c;整合 Nginx、HTTPS、LVS、Keepalived、MySQL 等服务&#xff0c;实现 Web 服务的负载均衡、高可用性及数据持久化。 案例场景&#xff1a;高可用 Web服务架构 目标 构建高可用 Web 集群&#xff0c;支持负载均衡和故障自动切换…

【QT】对话框dialog类封装

【QT】对话框dialog类封装 背景要点采用对输入框的信号监测实现端口和IP有效 实现 背景 在Qt 6.8.1 (MSVC 2022, x86_64)中进行编写&#xff0c;需要实现IP和端口号输入的弹窗&#xff0c;实现的方式有2种&#xff0c;其一&#xff0c;采用UI绘制&#xff0c;然后进行界面加载…

2025/5/26 学习日记 基本/扩展正则表达式 linux三剑客之grep

在 Linux 系统中&#xff0c;正则表达式&#xff08;Regular Expression可用于匹配、查找和替换符合特定模式的文本。根据语法和功能的不同&#xff0c;正则表达式可分为 基础正则表达式&#xff08;BRE&#xff09; 和 扩展正则表达式&#xff08;ERE&#xff09;。 基础正则…

【后端高阶面经:消息队列篇】29、Kafka高性能探秘:零拷贝、顺序写与分区并发实战

一、 顺序写入:磁盘性能的极致挖掘 Kafka的高性能本质上源于对磁盘顺序访问的深度优化。 传统随机写入的磁盘操作需要磁头频繁寻道,机械硬盘的随机写性能通常仅为100IOPS左右,而Kafka通过追加日志(Append-Only Log)模式,将所有消息按顺序写入分区文件,使磁盘操作转化为…

在 Docker 中启动 Jupyter Notebook

文章目录 一、创建容器二、Conda安装三、安装 Jupyter四、启动 Jupyter五、注册内核来使用虚拟环境小结 一、创建容器 可以先查看宿主机8888端口是否被占用&#xff0c;无输出&#xff0c;表明端口未被任何进程占用&#xff0c;如果有LISTEN&#xff0c;可能在创建容器的时候需…

CodeTop之LRU缓存

题目链接 146. LRU 缓存 - 力扣&#xff08;LeetCode&#xff09; 题目解析 算法原理 我们使用双向链表哈希表的形式来模拟缓存机制 首先我们要自己实现一个双链表, 自己写一个内部类, 这个内部类记录了key,value,prev,next(前驱和后继), 后续我们就通过这个内部类来构造双…

uboot常用命令之eMMC/SD卡命令

eMMC和SD卡(TF卡)是同一类设备&#xff0c;以下命令二者是通用&#xff0c;本章节主要以eMMC举例说明命令的使用。 使用help mmc可以看到mmc相关命令列表以及其对应命令用法&#xff1a; > help mmc 一、mmc dev 使用mmc list可以看到当前系统挂载的所有mmc设备&#xff…

知识宇宙-学习篇:学编程为什么从C语言开始学起?

名人说&#xff1a;博观而约取&#xff0c;厚积而薄发。——苏轼《稼说送张琥》 创作者&#xff1a;Code_流苏(CSDN)&#xff08;一个喜欢古诗词和编程的Coder&#x1f60a;&#xff09; 目录 一、C语言的历史地位与影响力1. 编程语言的"鼻祖"2. 现代技术的基础 二、…