C++設(shè)計(jì)模式之建造者模式
建造者模式
在GOF的《設(shè)計(jì)模式 可復(fù)用面向?qū)ο筌浖幕A(chǔ)》中是這樣說(shuō)的:將一個(gè)復(fù)雜對(duì)象的構(gòu)建與它的表示分離,使得同樣的構(gòu)建過(guò)程可以創(chuàng)建不同的表示。
這句話,似懂非懂的。一個(gè)復(fù)雜對(duì)象的創(chuàng)建,其通常是由很多的子對(duì)象構(gòu)成;如果一個(gè)對(duì)象能夠直接就創(chuàng)建好了,那么也不會(huì)稱之為復(fù)雜對(duì)象。由于項(xiàng)目中需求的變化,這個(gè)復(fù)雜對(duì)象的各個(gè)部分經(jīng)常會(huì)發(fā)生劇烈的變化,但是,不管怎么變化,將它們組合在一起,組成一個(gè)復(fù)雜的對(duì)象的事實(shí)是不會(huì)變的。建造者模式就提供了一種“封裝機(jī)制”來(lái)將各個(gè)對(duì)象的變化隔離開(kāi),最終,組合成復(fù)雜對(duì)象的過(guò)程是不會(huì)變的。
在《大話設(shè)計(jì)模式》一書(shū)中,例舉了一個(gè)很好的例子————建造小人。建造一個(gè)小人,要分為六步:頭部、身體、左手、右手、左腳和右腳。與抽象工廠模式不同的是,建造者模式是在Director的控制下一步一步的構(gòu)造出來(lái)的,在建造的過(guò)程中,建造者模式可以進(jìn)行更精細(xì)的控制。不管人的頭部、身體、左手、右手、左腳或者右腳如何變化,但是最終還是由這幾部分組合在一起形成一個(gè)人,雖然是同一個(gè)建造過(guò)程,但是這個(gè)人就會(huì)有不同的表示,比如,胖子,瘦子,個(gè)高的,個(gè)低的等等。
UML圖
類圖如下:
時(shí)序圖如下:
代碼實(shí)現(xiàn)
/*
** FileName : BuilderPattern
** Author : Jelly Young
** Date : 2013/11/22
** Description : More information, please go to //www.jb51.net
*/
#include <iostream>
using namespace std;
typedef enum MANTYPETag
{
kFatMan,
kThinMan,
kNormal
}MANTYPE;
class Man
{
public:
void SetHead(MANTYPE type){ m_Type = type; }
void SetBody(MANTYPE type){ m_Type = type; }
void SetLeftHand(MANTYPE type){ m_Type = type; }
void SetRightHand(MANTYPE type){ m_Type = type; }
void SetLeftFoot(MANTYPE type){ m_Type = type; }
void SetRightFoot(MANTYPE type){ m_Type = type; }
void ShowMan()
{
switch (m_Type)
{
case kFatMan:
cout<<"I'm a fat man"<<endl;
return;
case kThinMan:
cout<<"I'm a thin man"<<endl;
return;
default:
cout<<"I'm a normal man"<<endl;
return;
}
}
private:
MANTYPE m_Type;
};
// Builder
class Builder
{
public:
virtual void BuildHead(){}
virtual void BuildBody(){}
virtual void BuildLeftHand(){}
virtual void BuildRightHand(){}
virtual void BuildLeftFoot(){}
virtual void BuildRightFoot(){}
virtual Man *GetMan(){ return NULL; }
};
// FatManBuilder
class FatManBuilder : public Builder
{
public:
FatManBuilder(){ m_FatMan = new Man(); }
void BuildHead(){ m_FatMan->SetHead(kFatMan); }
void BuildBody(){ m_FatMan->SetBody(kFatMan); }
void BuildLeftHand(){ m_FatMan->SetLeftHand(kFatMan); }
void BuildRightHand(){ m_FatMan->SetRightHand(kFatMan); }
void BuildLeftFoot(){ m_FatMan->SetLeftFoot(kFatMan); }
void BuildRightFoot(){ m_FatMan->SetRightFoot(kFatMan); }
Man *GetMan(){ return m_FatMan; }
private:
Man *m_FatMan;
};
// ThisManBuilder
class ThinManBuilder : public Builder
{
public:
ThinManBuilder(){ m_ThinMan = new Man(); }
void BuildHead(){ m_ThinMan->SetHead(kThinMan); }
void BuildBody(){ m_ThinMan->SetBody(kThinMan); }
void BuildLeftHand(){ m_ThinMan->SetLeftHand(kThinMan); }
void BuildRightHand(){ m_ThinMan->SetRightHand(kThinMan); }
void BuildLeftFoot(){ m_ThinMan->SetLeftFoot(kThinMan); }
void BuildRightFoot(){ m_ThinMan->SetRightFoot(kThinMan); }
Man *GetMan(){ return m_ThinMan; }
private:
Man *m_ThinMan;
};
// Director
class Director
{
public:
Director(Builder *builder) { m_Builder = builder; }
void CreateMan();
private:
Builder *m_Builder;
};
void Director::CreateMan()
{
m_Builder->BuildHead();
m_Builder->BuildBody();
m_Builder->BuildLeftHand();
m_Builder->BuildRightHand();
m_Builder->BuildLeftHand();
m_Builder->BuildRightHand();
}
int main(int argc, char *argv[])
{
Builder *builderObj = new FatManBuilder();
Director directorObj(builderObj);
directorObj.CreateMan();
Man *manObj = builderObj->GetMan();
if (manObj == NULL)
return 0;
manObj->ShowMan();
delete builderObj;
builderObj = NULL;
return 0;
};
上面這個(gè)例子比較雜,但是也是建造者模式的應(yīng)用。下面這個(gè)例子是建造者最一般,最簡(jiǎn)單的實(shí)現(xiàn)方法:
/*
** FileName : BuilderPattern
** Author : Jelly Young
** Date : 2013/11/23
** Description : More information, please go to //www.jb51.net
*/
#include <iostream>
#include <vector>
using namespace std;
class Builder;
// Product
class Product
{
public:
void AddPart(const char *info) { m_PartInfoVec.push_back(info); }
void ShowProduct()
{
for (std::vector<const char *>::iterator item = m_PartInfoVec.begin();
item != m_PartInfoVec.end(); ++item)
{
cout<<*item<<endl;
}
}
private:
std::vector<const char *> m_PartInfoVec;
};
// Builder
class Builder
{
public:
virtual void BuildPartA() {}
virtual void BuildPartB() {}
virtual Product *GetProduct() { return NULL; }
};
// ConcreteBuilder
class ConcreteBuilder : public Builder
{
public:
ConcreteBuilder() { m_Product = new Product(); }
void BuildPartA()
{
m_Product->AddPart("PartA completed");
}
void BuildPartB()
{
m_Product->AddPart("PartB completed");
}
Product *GetProduct() { return m_Product; }
private:
Product *m_Product;
};
// Director
class Director
{
public:
Director(Builder *builder) { m_Builder = builder; }
void CreateProduct()
{
m_Builder->BuildPartA();
m_Builder->BuildPartB();
}
private:
Builder *m_Builder;
};
// main
int main()
{
Builder *builderObj = new ConcreteBuilder();
Director directorObj(builderObj);
directorObj.CreateProduct();
Product *productObj = builderObj->GetProduct();
if (productObj == NULL)
{
return 0;
}
productObj->ShowProduct();
delete builderObj;
builderObj = NULL;
}
通過(guò)比較上面的兩個(gè)例子,可以很容易的把建造者模式的骨架抽象出來(lái)。
使用要點(diǎn)
1.建造者模式生成的對(duì)象有復(fù)雜的內(nèi)部結(jié)構(gòu),將分步驟的去構(gòu)建一個(gè)復(fù)雜的對(duì)象,分多少步是確定的,而每一步的實(shí)現(xiàn)是不同的,可能經(jīng)常發(fā)生變化;
2.在上面的例子中,我們都看到了最終生成的Man和Product都沒(méi)有抽象類,這又導(dǎo)出建造者適用的一種情況,當(dāng)需要?jiǎng)?chuàng)建復(fù)雜對(duì)象的過(guò)程中,復(fù)雜對(duì)象沒(méi)有多少共同的特點(diǎn),很難抽象出來(lái)時(shí),而復(fù)雜對(duì)象的組裝又有一定的相似點(diǎn)時(shí),建造者模式就可以發(fā)揮出作用。簡(jiǎn)單的說(shuō),可能使用了建造者模式,最終建造的對(duì)象可能沒(méi)有多大的關(guān)系,關(guān)于這一點(diǎn),閱讀《設(shè)計(jì)模式 可復(fù)用面向?qū)ο筌浖幕A(chǔ)》中的建造者模式時(shí)是最有體會(huì)的。
總結(jié)
一個(gè)復(fù)雜對(duì)象是由多個(gè)部件組成的,建造者模式是把復(fù)雜對(duì)象的創(chuàng)建和部件的創(chuàng)建分別開(kāi)來(lái),分別用Builder類和Director類來(lái)表示。用Director構(gòu)建最后的復(fù)雜對(duì)象,而在上面Builder接口中封裝的是如何創(chuàng)建一個(gè)個(gè)部件(復(fù)雜對(duì)象是由這些部件組成的),也就是說(shuō),Director負(fù)責(zé)如何將部件最后組裝成產(chǎn)品。這樣建造者模式就讓設(shè)計(jì)和實(shí)現(xiàn)解耦了。
剛開(kāi)始接觸建造者模式的時(shí)候,最容易把建造者和抽象工廠模式混淆了。由于而這都屬于創(chuàng)建型的設(shè)計(jì)模式,所以二者之間是有公共點(diǎn)的,但是建造者模式注重于對(duì)象組合,即不同的小對(duì)象組成一個(gè)整體的復(fù)雜大對(duì)象,而抽象工廠模式針對(duì)于接口編程,只是對(duì)外提供創(chuàng)建對(duì)象的工廠接口,不負(fù)責(zé)對(duì)象之后的處理。
建造者模式,是一個(gè)比較復(fù)雜,不容易權(quán)衡的設(shè)計(jì)模式。大家應(yīng)該更多的閱讀開(kāi)源代碼,理解他人是如何使用該模式的。從實(shí)際的應(yīng)用中學(xué)習(xí)設(shè)計(jì)模式。
上一篇:C++二進(jìn)制翻轉(zhuǎn)實(shí)例分析
欄 目:C語(yǔ)言
下一篇:C語(yǔ)言實(shí)現(xiàn)兩個(gè)遞減數(shù)列中尋找某一個(gè)數(shù)
本文標(biāo)題:C++設(shè)計(jì)模式之建造者模式
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/3343.html
您可能感興趣的文章
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 01-10深入理解C++中常見(jiàn)的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10APUE筆記之:進(jìn)程環(huán)境詳解
- 01-10c++中inline的用法分析
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解
- 01-10深入C/C++浮點(diǎn)數(shù)在內(nèi)存中的存儲(chǔ)方式詳解


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 04-02jquery與jsp,用jquery
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子