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

相关文章

Python使用getopt处理命令行参数示例解析(最佳实践)

《Python使用getopt处理命令行参数示例解析(最佳实践)》getopt模块是Python标准库中一个简单但强大的命令行参数处理工具,它特别适合那些需要快速实现基本命令行参数解析的场景,或者需要... 目录为什么需要处理命令行参数?getopt模块基础实际应用示例与其他参数处理方式的比较常见问http

python实现svg图片转换为png和gif

《python实现svg图片转换为png和gif》这篇文章主要为大家详细介绍了python如何实现将svg图片格式转换为png和gif,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录python实现svg图片转换为png和gifpython实现图片格式之间的相互转换延展:基于Py

Python利用ElementTree实现快速解析XML文件

《Python利用ElementTree实现快速解析XML文件》ElementTree是Python标准库的一部分,而且是Python标准库中用于解析和操作XML数据的模块,下面小编就来和大家详细讲讲... 目录一、XML文件解析到底有多重要二、ElementTree快速入门1. 加载XML的两种方式2.

C 语言中enum枚举的定义和使用小结

《C语言中enum枚举的定义和使用小结》在C语言里,enum(枚举)是一种用户自定义的数据类型,它能够让你创建一组具名的整数常量,下面我会从定义、使用、特性等方面详细介绍enum,感兴趣的朋友一起看... 目录1、引言2、基本定义3、定义枚举变量4、自定义枚举常量的值5、枚举与switch语句结合使用6、枚

Nginx location匹配模式与规则详解

《Nginxlocation匹配模式与规则详解》:本文主要介绍Nginxlocation匹配模式与规则,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、环境二、匹配模式1. 精准模式2. 前缀模式(不继续匹配正则)3. 前缀模式(继续匹配正则)4. 正则模式(大

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

使用Python从PPT文档中提取图片和图片信息(如坐标、宽度和高度等)

《使用Python从PPT文档中提取图片和图片信息(如坐标、宽度和高度等)》PPT是一种高效的信息展示工具,广泛应用于教育、商务和设计等多个领域,PPT文档中常常包含丰富的图片内容,这些图片不仅提升了... 目录一、引言二、环境与工具三、python 提取PPT背景图片3.1 提取幻灯片背景图片3.2 提取

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Python实现图片分割的多种方法总结

《Python实现图片分割的多种方法总结》图片分割是图像处理中的一个重要任务,它的目标是将图像划分为多个区域或者对象,本文为大家整理了一些常用的分割方法,大家可以根据需求自行选择... 目录1. 基于传统图像处理的分割方法(1) 使用固定阈值分割图片(2) 自适应阈值分割(3) 使用图像边缘检测分割(4)

Android实现在线预览office文档的示例详解

《Android实现在线预览office文档的示例详解》在移动端展示在线Office文档(如Word、Excel、PPT)是一项常见需求,这篇文章为大家重点介绍了两种方案的实现方法,希望对大家有一定的... 目录一、项目概述二、相关技术知识三、实现思路3.1 方案一:WebView + Office Onl