C++设计模式-创建型设计模式:构建者(Builder)

目录

Builder构建者模式:场景、功能、组成部分

场景

功能

组成部分:Builer+Director


Builder构建者模式:场景、功能、组成部分

场景

 还是老套路,首先不用设计模式来实现这个功能:

#include <iostream>
#include <string>
#include <vector>
#include <sstream>
using namespace std;

template<class T>
string ConvertToString(T value) {
	stringstream ss;
	ss << value;
	return ss.str();
}

class ExportHeaderModel {
public:
	ExportHeaderModel(string strDepId, string strExportDate)
		:m_strDepId(strDepId), m_strExportDate(strExportDate)
	{}
	string getStrDepId() {
		return m_strDepId;
	}
	string getStrExportDate() {
		return m_strExportDate;
	}
private:
	string m_strDepId;//对账单的部门id
	string m_strExportDate;//对账单的导出日期
};

class ExportDataModel {
public:
	ExportDataModel(string strTransId,double Quantity)
		:m_strTransId(strTransId), m_Quantity(Quantity)
	{}
	string getStrTransId() { return m_strTransId; }
	double getQuantity() { return m_Quantity; }
private:
	string m_strTransId;//交易ID
	double m_Quantity;
};

class ExportFooterModel {
public:
	ExportFooterModel(string exportUser)
		:m_exportUser(exportUser)
	{}
	string getExportUser() {
		return m_exportUser;
	}
private:
	string m_exportUser;
};

//考虑一个周期,有多笔交易记录
class ExportToTextHelper {
public:
	void doExport(ExportHeaderModel& ehm, vector<ExportDataModel*>edmCollection, ExportFooterModel& efm) {
		string strTemp = "";

		//第一步:写文件头
		strTemp += ehm.getStrDepId() + "," + ehm.getStrExportDate() + "\n";
		//第二部:写文件体数据
		for (auto it = edmCollection.begin(); it != edmCollection.end(); it++) {
			strTemp += (*it)->getStrTransId() + ":" + ConvertToString((*it)->getQuantity()) + "\n";
		}
		//第三步:拼接文件尾
		strTemp += efm.getExportUser() + "\n";
		cout << strTemp << endl;
	}
};

class ExportToXmlHelper {
public:
	void doExport(ExportHeaderModel& ehm, vector<ExportDataModel*>edmCollection, ExportFooterModel& efm) {
		string strTemp = "";

		//第一步:拼接文件头
		strTemp.append("<?xml version='1.0' enconding='utf-8'>\n ");
		strTemp.append("<Recipt>\n");
		strTemp.append("    <Header>\n");
		strTemp.append("        <DepId>");
		strTemp.append(ehm.getStrDepId() + "</DepId>\n");
		strTemp.append("        <ExportData>" + ehm.getStrExportDate() + "</ExportData>\n");
		strTemp.append("    </Header>\n");
		//第二部:构建数据体
		strTemp.append("    <Body>\n");
		for (auto it = edmCollection.begin(); it != edmCollection.end(); it++) {
			strTemp.append("        <id>" + (*it)->getStrTransId() + "</id>\n");
			strTemp.append("        <amount>" + ConvertToString((*it)->getQuantity()) + "</amount>\n");
		}
		strTemp.append("    </Body>\n");
		//第三步:拼接文件尾
		strTemp.append("    <Footer>\n");
		strTemp.append("        <ExportUser>" + efm.getExportUser() + "</ExportUser>\n");
		strTemp.append("    </Footer>\n</Recipt>\n");

		cout << strTemp << endl;
	}
};

int main() {
	ExportHeaderModel* pEhm = new ExportHeaderModel("南京1支行", "1st June");

	ExportDataModel* pEdm1 = new ExportDataModel("1", 10000.0f);
	ExportDataModel* pEdm2 = new ExportDataModel("2", 20000.0f);
	vector<ExportDataModel*>Vec;
	Vec.push_back(pEdm1);
	Vec.push_back(pEdm2);
	
	ExportFooterModel* pEfm = new ExportFooterModel("yxh");

	ExportToTextHelper helper1;
	helper1.doExport(*pEhm, Vec, *pEfm);

	ExportToXmlHelper helper2;
	helper2.doExport(*pEhm, Vec, *pEfm);

	return 0;
}

不错,但有一个问题,两种Helper中都存在着相同的步骤一二三。

既然都学了设计模式了,这种重复性的代码必然是不符合高逼格的,肯定要重构!

这里就引出了Builder模式

#include <iostream>
#include <string>
#include <vector>
#include <sstream>
using namespace std;

template<class T>
string ConvertToString(T value) {
	stringstream ss;
	ss << value;
	return ss.str();
}

class ExportHeaderModel {
public:
	ExportHeaderModel(string strDepId, string strExportDate)
		:m_strDepId(strDepId), m_strExportDate(strExportDate)
	{}
	string getStrDepId() {
		return m_strDepId;
	}
	string getStrExportDate() {
		return m_strExportDate;
	}
private:
	string m_strDepId;//对账单的部门id
	string m_strExportDate;//对账单的导出日期
};

class ExportDataModel {
public:
	ExportDataModel(string strTransId, double Quantity)
		:m_strTransId(strTransId), m_Quantity(Quantity)
	{}
	string getStrTransId() { return m_strTransId; }
	double getQuantity() { return m_Quantity; }
private:
	string m_strTransId;//交易ID
	double m_Quantity;
};

class ExportFooterModel {
public:
	ExportFooterModel(string exportUser)
		:m_exportUser(exportUser)
	{}
	string getExportUser() {
		return m_exportUser;
	}
private:
	string m_exportUser;
};

//创建者模式
class Builder {
public:
	virtual void builderHeader(ExportHeaderModel& ehm) = 0;
	virtual void builderBody(vector<ExportDataModel*>& edmCollection) = 0;
	virtual void builderFoot(ExportFooterModel& efm) = 0;
	virtual string getResult() = 0;
protected:
	Builder() {}
};

class TextBuilder :public Builder {
public:
	void builderHeader(ExportHeaderModel& ehm) {
		m_strResult.append(ehm.getStrDepId() + "," + ehm.getStrExportDate() + "\n");
	}
	void builderBody(vector<ExportDataModel*>& edmCollection) {
		for (auto it = edmCollection.begin(); it != edmCollection.end(); it++) {
			m_strResult += (*it)->getStrTransId() + ":" + ConvertToString((*it)->getQuantity()) + "\n";
		}
	}
	void builderFoot(ExportFooterModel& efm) {
		m_strResult += efm.getExportUser() + "\n";
	}
	string getResult() {
		return m_strResult;
	}
public:
	TextBuilder() { m_strResult = ""; }
private:
	string m_strResult;
};


class Director {
public:
	Director(Builder* pBuilder) :m_pBuilder(pBuilder) {}
	void construct(ExportHeaderModel& ehm, vector<ExportDataModel*>edmCollection, ExportFooterModel& efm) {
		m_pBuilder->builderHeader(ehm);
		m_pBuilder->builderBody(edmCollection);
		m_pBuilder->builderFoot(efm);
	}
private:
	Builder* m_pBuilder;
};

int main() {
	ExportHeaderModel* pEhm = new ExportHeaderModel("南京1支行", "1st June");

	ExportDataModel* pEdm1 = new ExportDataModel("1", 10000.0f);
	ExportDataModel* pEdm2 = new ExportDataModel("2", 20000.0f);
	vector<ExportDataModel*>Vec;
	Vec.push_back(pEdm1);
	Vec.push_back(pEdm2);

	ExportFooterModel* pEfm = new ExportFooterModel("yxh");


	Builder* pBuilder = new TextBuilder();
	Director* pDirector = new Director(pBuilder); 
	pDirector->construct(*pEhm, Vec, *pEfm);
	cout << pBuilder->getResult() << endl;
	return 0;
}

这里只重构了Text的创建。

其实总体看来,设计模式的目的就是:把不变的给客户,把以后可能变化的留给开发者,并且要便于开发者增加功能模块,要达到模块于模块之间的松耦合。

功能

  • 构建复杂产品,而且是细化的,分步骤的构建产品,重在解决一步一步构造复杂对象的问题
  • 这个构建过程是统一的,固定不变的,变化的部分就放到生成器Builder中
  • 只要配置不同的Builder,那么同样的构建过程,就能构建出不同的产品出来

组成部分:Builer+Director

  • Builder接口,定义如何构建各个部件,每个部件功能如何实现,如何装配这些部件到产品中去
  • Director,定义如何组合来构建产品,负责整体的构建算法,通常分步骤来执行

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Builder模式 将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示 统设计中,有时候面临着一个“复杂系统”的创建工作,该对象通常由各个部分的子对象用一定的算法构成,或者说按一定的步骤组合而成;这些的算法和步骤是稳定的,而构成这个对象的子对象却经常由于需求改变而发生变化,此时我们可以选择使用Builder模式。尽管Builder模式相当容易让人误解,但我认为这几点应该是不会有错的。因此我个人认为Builder模式中要好好把握如下几点 1. 需要创建的对象是复杂对象 2. 对象的创建过程是一步步创建(即:可能由于过程的不同而导致有不同的对象展示) 3. 明确建造者(Builder)、具体建造者(Concrete Builder)、指导者(Director)、产品(Product)之间的职责和联系。 ◆建造者(Builder): 给出一个抽象接口,以规范产品对象的各个组成成分的建造。一般而言,此接口独立于应用程序的商业逻辑。模式中直接创建产品对象的是 具体建造者(ConcreteBuilder):具体建造者类必须实现这个接口所要求的方法:一个是建造方法,另一个是结果返还方法。 ◆具体建造者(Concrete Builder): 担任这个角色的是于应用程序紧密相关的类,它们在应用程序调用下创建产品实例。这个角色主要完成的任务包括: ■实现Builder角色提供的接口,一步一步完成创建产品实例的过程。 ■在建造过程完成后,提供产品的实例。 ◆指导者(Director): 担任这个角色的类调用具体建造者角色以创建产品对象。导演者并没有产品类的具体知识,真正拥有产品类的具体知识的是具体建造者对象。 ◆产品(Product): 产品便是建造中的复杂对象。 详细见博客 http://blog.csdn.net/xiaoting451292510/article/details/8330462
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值