C++使用内部类实现一种Builder模式(属性内容不对外暴露, 不需管理指针)

本文主要是介绍C++使用内部类实现一种Builder模式(属性内容不对外暴露, 不需管理指针),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

假设我们有一个生产车的工厂,每辆车有如下属性: 系列、颜色、载人数

头文件如下:

#ifndef TEST_H
#define TEST_H#include <string>
#include <memory>using namespace std;class Car
{
public:class Builder{public:Builder(const Builder &builderArg);Builder(const Builder &&builderArg);Builder& operator=(const Builder &builderArg);Builder& operator=(const Builder &&builderArg);Builder(const string &series);Builder& setColor(const string &color);    Builder& setSize(const int size);                    Car build();private:shared_ptr<void> carBuildData;friend class Car;};void drive();Car() = default;~Car() = default;Car(const Builder &builder);Car(const Car &carArg);Car(const Car &&carArg);Car& operator=(const Car &carArg);Car& operator=(const Car &&carArg);private:shared_ptr<void> carData;
};#endif // !TEST_H

shared_ptr<void>的目的在于希望内部数据隐藏,对外不体现。

实现如下:

#include "test.h"
#include <iostream>
#include <functional>struct CarData_t{string series;          //  系列string color = "RED";   //  颜色int size = 5;           //  大小(载人数)CarData_t(const string &seriesArg) : series(seriesArg) {}
};Car::Builder::Builder(const Car::Builder &builderArg)
{carBuildData = builderArg.carBuildData;
}Car::Builder::Builder(const Car::Builder &&builderArg)
{carBuildData = builderArg.carBuildData;
}Car::Builder& Car::Builder::operator=(const Car::Builder &builderArg)
{carBuildData = builderArg.carBuildData;return *this;
}Car::Builder& Car::Builder::operator=(const Car::Builder &&builderArg)
{carBuildData = builderArg.carBuildData;return *this;
}Car::Car(const Car &carArg)
{carData = carArg.carData;
}Car::Car(const Car &&carArg)
{carData = carArg.carData;
}Car& Car::operator=(const Car &carArg)
{carData = carArg.carData;return *this;
}Car& Car::operator=(const Car &&carArg)
{carData = carArg.carData;return *this;
}Car::Builder::Builder(const string &series)
{CarData_t testdata(series);carBuildData = make_shared<CarData_t>(testdata);
}Car Car::Builder::build()
{Car test(*this);// 这里直接返回, 函数返回值Car未加&,会触发移动构造函数,如果返回值为Car&,会触发拷贝构造函数return test;
}//  减少重复造轮子,每个set函数都要判断指针是否为空,用回调可以避免重复判断
void operateTestDataArg(void* testDataVoidPtr, function<void(CarData_t*)> callback)
{if (testDataVoidPtr == nullptr) { return; }auto testDataPtr = reinterpret_cast<CarData_t*>(testDataVoidPtr);if (testDataPtr == nullptr) {return;}callback(testDataPtr);
}void Car::drive()
{operateTestDataArg(carData.get(), [&](CarData_t* testDataPtr) {cout << "series : " << testDataPtr->series << "    ";cout << "color : " << testDataPtr->color << "    ";cout << "size : " << testDataPtr->size << endl;});
}Car::Car(const Builder &builder)
{operateTestDataArg(builder.carBuildData.get(), [&](CarData_t* buildDataPtr) {carData = make_shared<CarData_t>(*buildDataPtr);});
}Car::Builder& Car::Builder::setColor(const string &color)
{operateTestDataArg(carBuildData.get(), [&](CarData_t* buildDataPtr) {buildDataPtr->color = color;});return *this;
}Car::Builder& Car::Builder::setSize(const int size)
{operateTestDataArg(carBuildData.get(), [&](CarData_t* buildDataPtr) {buildDataPtr->size = size;});return *this;
}

下面来使用:

int main()
{auto benz = Car::Builder("Benz").build();cout << "1 : Benz              "; benz.drive();//  设置benz为公司用车,由于实现了赋值运算符函数,使用了shared_ptr, benz析构后CompanyCar也能继续使用,CompanyCar和benz中的carData指向同一片空间,对于shared_ptr来说,指向该空间的指针均析构了后该空间才会自动释放auto CompanyCar = benz;cout << "2 : CompanyCar        "; CompanyCar.drive();auto HondaBaseBuilder = Car::Builder("Honda");Car HondaDefault(HondaBaseBuilder);cout << "3 : HondaDefault      "; HondaDefault.drive();auto HondaBlue = HondaBaseBuilder.setColor("Blue").setSize(5).build();cout << "4 : HondaBlue         "; HondaBlue.drive();auto HondaSmallBlack = HondaBaseBuilder.setColor("Black").setSize(2).build();cout << "5 : HondaSmallBlack   "; HondaSmallBlack.drive();//  ps:HondaBaseBuilder属性一直在变 此时初始化car已经不是默认值了,可以自己加入reset函数扩展 或者重新设置属性,此处功能为保证为Honda系列,不用重复Builder("Honda")Car HondaCurrent(HondaBaseBuilder);cout << "5 : HondaCurrent      "; HondaCurrent.drive();return 0;
}

运行结果:

1 : Benz              series : Benz     color : RED    size : 5
2 : CompanyCar        series : Benz     color : RED    size : 5
3 : HondaDefault      series : Honda    color : RED    size : 5
4 : HondaBlue         series : Honda    color : Blue   size : 5
5 : HondaSmallBlack   series : Honda    color : Black  size : 2
5 : HondaCurrent      series : Honda    color : Black  size : 2

这篇关于C++使用内部类实现一种Builder模式(属性内容不对外暴露, 不需管理指针)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

LiteFlow轻量级工作流引擎使用示例详解

《LiteFlow轻量级工作流引擎使用示例详解》:本文主要介绍LiteFlow是一个灵活、简洁且轻量的工作流引擎,适合用于中小型项目和微服务架构中的流程编排,本文给大家介绍LiteFlow轻量级工... 目录1. LiteFlow 主要特点2. 工作流定义方式3. LiteFlow 流程示例4. LiteF

使用Python开发一个现代化屏幕取色器

《使用Python开发一个现代化屏幕取色器》在UI设计、网页开发等场景中,颜色拾取是高频需求,:本文主要介绍如何使用Python开发一个现代化屏幕取色器,有需要的小伙伴可以参考一下... 目录一、项目概述二、核心功能解析2.1 实时颜色追踪2.2 智能颜色显示三、效果展示四、实现步骤详解4.1 环境配置4.

canal实现mysql数据同步的详细过程

《canal实现mysql数据同步的详细过程》:本文主要介绍canal实现mysql数据同步的详细过程,本文通过实例图文相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的... 目录1、canal下载2、mysql同步用户创建和授权3、canal admin安装和启动4、canal

CSS3中的字体及相关属性详解

《CSS3中的字体及相关属性详解》:本文主要介绍了CSS3中的字体及相关属性,详细内容请阅读本文,希望能对你有所帮助... 字体网页字体的三个来源:用户机器上安装的字体,放心使用。保存在第三方网站上的字体,例如Typekit和Google,可以link标签链接到你的页面上。保存在你自己Web服务器上的字

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔

Nexus安装和启动的实现教程

《Nexus安装和启动的实现教程》:本文主要介绍Nexus安装和启动的实现教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、Nexus下载二、Nexus安装和启动三、关闭Nexus总结一、Nexus下载官方下载链接:DownloadWindows系统根

SQL中JOIN操作的条件使用总结与实践

《SQL中JOIN操作的条件使用总结与实践》在SQL查询中,JOIN操作是多表关联的核心工具,本文将从原理,场景和最佳实践三个方面总结JOIN条件的使用规则,希望可以帮助开发者精准控制查询逻辑... 目录一、ON与WHERE的本质区别二、场景化条件使用规则三、最佳实践建议1.优先使用ON条件2.WHERE用

SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程

《SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程》LiteFlow是一款专注于逻辑驱动流程编排的轻量级框架,它以组件化方式快速构建和执行业务流程,有效解耦复杂业务逻辑,下面给大... 目录一、基础概念1.1 组件(Component)1.2 规则(Rule)1.3 上下文(Conte

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma

MySQL 衍生表(Derived Tables)的使用

《MySQL衍生表(DerivedTables)的使用》本文主要介绍了MySQL衍生表(DerivedTables)的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学... 目录一、衍生表简介1.1 衍生表基本用法1.2 自定义列名1.3 衍生表的局限在SQL的查询语句select