青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

積木

No sub title

  C++博客 :: 首頁 :: 聯(lián)系 :: 聚合  :: 管理
  140 Posts :: 1 Stories :: 11 Comments :: 0 Trackbacks

常用鏈接

留言簿(1)

我參與的團(tuán)隊(duì)

搜索

  •  

最新評(píng)論

閱讀排行榜

評(píng)論排行榜

轉(zhuǎn)載自:http://patmusing.blog.163.com/blog/static/13583496020101502024824/


也稱為
Objects for States模式。

允許一個(gè)對(duì)象在其內(nèi)部狀態(tài)改變時(shí)改變其行為,從而使對(duì)象看起來似乎修改了類。

“Allow an object to alter its behavior when its internal state changes. The object will appear to change its class.” – GoF

一個(gè)應(yīng)用的行為往往會(huì)根據(jù)其內(nèi)部變量值而表現(xiàn)出很大的不同。比如用Visual Studio開發(fā)程序,當(dāng)寫了一段程序后,如果點(diǎn)擊保存的按鈕,則所寫的程序便保存到了硬盤上,然后保存按鈕變灰,這時(shí)候程序文件時(shí)saved的狀態(tài);在這個(gè)狀態(tài)上,如果有加了幾行代碼,那么狀態(tài)就變成unsaved,此時(shí),保存按鈕又可以點(diǎn)擊。savedunsaved是程序文件的狀態(tài),保存這個(gè)動(dòng)作是行為,對(duì)于saved或者unsaved,同樣的保存動(dòng)作有不同的意義,即,如果程序文件時(shí)saved的狀態(tài),那么點(diǎn)擊保存按鈕,則什么也不做(按鈕式灰色的);如果程序文件時(shí)unsaved狀態(tài),點(diǎn)擊保存按鈕,則保存程序文件。這個(gè)案例說明,對(duì)象的狀態(tài)和其行為有密切的關(guān)系。

要解決類似上面的問題,通常的做法:

1. 設(shè)置一個(gè)狀態(tài)變量

2. switch – case或者if – else – if這樣的辦法來判定對(duì)象的狀態(tài),然后根據(jù)所得到的狀態(tài),選擇不同行為

如果增加了一個(gè)狀態(tài),那么

1. 增加一個(gè)狀態(tài)變量

2. 修改switch – case或者if – else – if語句,這顯然不符合OCP的原則(這里會(huì)增加編譯的時(shí)間成本),也沒有把程序變化部分和相對(duì)穩(wěn)定的部分隔離開,這會(huì)削弱應(yīng)用程序的可擴(kuò)展性

State模式就是為了解決上面的問題而產(chǎn)生的。即專門處理當(dāng)對(duì)象處于不同的狀態(tài)時(shí),將會(huì)有不同的行為這樣的情況。

State模式的UML類圖如下:

23. C++實(shí)現(xiàn)Behavioral - State模式 - 玄機(jī)逸士 - 玄機(jī)逸士博客

考慮一個(gè)門的狀態(tài):

23. C++實(shí)現(xiàn)Behavioral - State模式 - 玄機(jī)逸士 - 玄機(jī)逸士博客

說明:

如果一個(gè)門處于Opened狀態(tài),那么對(duì)其施加close動(dòng)作,就會(huì)使門的狀態(tài)變成Closed的狀態(tài);

如果一個(gè)門處于Opened狀態(tài),那么對(duì)其施加open動(dòng)作,那么門的狀態(tài)不會(huì)改變,依舊是Opened的狀態(tài);

如果一個(gè)門處于Closed狀態(tài),那么對(duì)其施加open動(dòng)作,就會(huì)使門的狀態(tài)變成Opened的狀態(tài);

如果一個(gè)門處于Closed狀態(tài),那么對(duì)其施加close動(dòng)作,那么門的狀態(tài)不會(huì)改變,依舊是Closed的狀態(tài);

下面用State模式具體實(shí)現(xiàn)上面業(yè)務(wù)的C++代碼:

// State.h

#include <iostream>

#include <string>

#include <typeinfo>

#include <memory>

using namespace std;

// 前置聲明

class Door;

// 門的狀態(tài)

class DoorState

{

public:

virtual ~DoorState();

public:

virtual void open(Door* door) = 0; // 開門

virtual void close(Door* door) = 0; // 關(guān)門

};

// 打開狀態(tài)的門

class DoorOpened : public DoorState

{

public:

~DoorOpened();

public:

void open(Door* door);

void close(Door* door);

};

// 關(guān)閉狀態(tài)的門

class DoorClosed : public DoorState

{

public:

~DoorClosed();

public:

void open(Door* door);

void close(Door* door);

};

//

class Door

{

private:

auto_ptr<DoorState> door_state; // 門的狀態(tài)

public:

Door();

~Door();

public:

void open(); // 開門

void close(); // 關(guān)門

void set_state(auto_ptr<DoorState> door_state); // 設(shè)置門的狀態(tài)

string get_state(); // 獲取門的狀態(tài)

};

// State.cpp

#include "State.h"

DoorState::~DoorState()

{

cout << "in the destructor of DoorState..." << endl;

}

DoorOpened::~DoorOpened()

{

cout << "in the destructor of DoorOpened..." << endl;

}

// 如果門已經(jīng)是打開的狀態(tài),則僅輸出相關(guān)信息

void DoorOpened::open(Door* door)

{

cout << "The Door is already opened. State will not change..." << endl;

}

// 如果門是關(guān)閉的狀態(tài),則關(guān)門

void DoorOpened::close(Door* door)

{

auto_ptr<DoorState> ds(new DoorClosed());

door->set_state(ds);

}

DoorClosed::~DoorClosed()

{

cout << "in the destructor of DoorClosed..." << endl;

}

// 如果門是關(guān)閉的狀態(tài),則開門

void DoorClosed::open(Door* door)

{

auto_ptr<DoorState> ds(new DoorOpened());

door->set_state(ds);

}

// 如果門已經(jīng)是關(guān)閉的狀態(tài),則僅輸出相關(guān)信息

void DoorClosed::close(Door* door)

{

cout << "The Door is already closed. State will not change..." << endl;

}

Door::Door()

{

auto_ptr<DoorState> ds(new DoorOpened()); // 設(shè)定門的初始狀態(tài)

door_state = ds;

}

Door::~Door()

{

cout << "in the destructor of Door..." << endl;

}

void Door::open() // 開門

{

door_state->open(this);

}

void Door::close() // 關(guān)門

{

door_state->close(this);

}

void Door::set_state(auto_ptr<DoorState> door_state) // 設(shè)定門的狀態(tài)

{

this->door_state = door_state;

}

// 利用RTTI技術(shù),輸出門的狀態(tài)

string Door::get_state()

{ // *(door_state.get())*(door_state)的寫法是等價(jià)的,get()表示獲取auto_ptr對(duì)象中underlying對(duì)象的真實(shí)指針,因此語

// 法上顯得更好理解一些。但*auto_ptr中也被重載過了,因此兩者的結(jié)果是一樣的。

string temp = typeid(*(door_state.get())).name(); // temp將等于 "class DoorOpened""class DoorClosed"

size_t found = temp.find("Door"); // 找到"Door"temp中位置

temp = temp.substr(found + 4); // 去掉"Door"(包括其自身)前面的字符串

return temp;

}

// PatternClient.cpp

#include "State.h"

int main(int argc, char** argv)

{

Door door;

cout << "1. Initial state of the door: " << door.get_state() << endl;

cout << "------------------------------------------------------" << endl;

cout << endl;

door.open();

cout << "2. State after opening the door: " << door.get_state() << endl;

cout << "------------------------------------------------------" << endl;

cout << endl;

door.close();

cout << "3. State after closing the door: " << door.get_state() << endl;

cout << "------------------------------------------------------" << endl;

cout << endl;

door.close();

cout << "4. State after closing the door: " << door.get_state() << endl;

cout << "------------------------------------------------------" << endl;

cout << endl;

door.open();

cout << "5. State after opening the door: " << door.get_state() << endl;

cout << "------------------------------------------------------" << endl;

cout << endl;

return 0;

}

輸出結(jié)果:

1. Initial state of the door: Opened

------------------------------------------------------

The Door is already opened. State will not change...

2. State after opening the door: Opened

------------------------------------------------------

in the destructor of DoorOpened...

in the destructor of DoorState...

3. State after closing the door: Closed

------------------------------------------------------

The Door is already closed. State will not change...

4. State after closing the door: Closed

------------------------------------------------------

in the destructor of DoorClosed...

in the destructor of DoorState...

5. State after opening the door: Opened

------------------------------------------------------

in the destructor of Door...

in the destructor of DoorOpened...

in the destructor of DoorState...

上面程序的UML類圖如下:

23. C++實(shí)現(xiàn)Behavioral - State模式 - 玄機(jī)逸士 - 玄機(jī)逸士博客

要點(diǎn):

- State設(shè)計(jì)模式將一個(gè)特定狀態(tài)相關(guān)的行為都放入一個(gè)State的子類對(duì)象中,在對(duì)象狀態(tài)切換時(shí),切換相應(yīng)的對(duì)象;但同時(shí)維持State的接口穩(wěn)定,這樣實(shí)現(xiàn)了具體操作與狀態(tài)轉(zhuǎn)換之間的解耦。

- 為不同的狀態(tài)引入不同的對(duì)象,使得一個(gè)狀態(tài)和一個(gè)對(duì)象一一對(duì)應(yīng),這也是為什么這個(gè)模式也稱為Objects for States的原因。這使得狀態(tài)轉(zhuǎn)換變得更加明確,而且可以保證不會(huì)出現(xiàn)狀態(tài)不一致的情況,因?yàn)閷?duì)象的狀態(tài)是由一個(gè)類來表示的,那么就保證了狀態(tài)轉(zhuǎn)換的原子性 即要么轉(zhuǎn)換成功,要不轉(zhuǎn)換 (即上面示例程序中的door_state要么被設(shè)定為DoorOpened,要么被設(shè)定為DoorClosed。如果一個(gè)對(duì)象有很多狀態(tài),采用通常的方法會(huì)有很復(fù)雜的if-else if-else語句,維護(hù)這些復(fù)雜的語句,很有可能導(dǎo)致各種麻煩,其中之一就是狀態(tài)的不一致。)

- State模式和Strategy模式很類似。它們最主要的區(qū)別就是:State模式是用不同的對(duì)象去封裝不同的狀態(tài),而Stategy模式用不同的對(duì)象去封裝不同的算法。

參考:

1. auto_ptr的用途和原理,詳見:http://patmusing.blog.163.com/blog/static/13583496020101824142699/

2. 使用auto_ptr的注意事項(xiàng),詳見:http://patmusing.blog.163.com/blog/static/13583496020101824541270/

3. 關(guān)于RTTI,詳見Stan Lippman的《C++ Primer》第4版之第18



posted on 2013-03-08 15:24 Jacc.Kim 閱讀(287) 評(píng)論(0)  編輯 收藏 引用 所屬分類: 設(shè)計(jì)模式
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲综合国产精品| 欧美韩日一区| 欧美伊人久久久久久午夜久久久久 | 国产精品永久免费视频| 亚洲精品一区二区三| 久久久久网站| 欧美一区二区在线视频| 国产欧美一区二区精品婷婷| 亚洲尤物在线视频观看| 99精品视频免费全部在线| 欧美激情精品久久久久久免费印度 | 亚洲经典一区| 午夜日韩av| 国产欧美午夜| 久久久www成人免费毛片麻豆| 亚洲小视频在线观看| 国产精品日本精品| 久久久久国色av免费观看性色| 亚洲欧美日韩精品久久亚洲区 | 亚洲激情另类| 亚洲韩国一区二区三区| 欧美高清视频免费观看| 洋洋av久久久久久久一区| 亚洲免费观看视频| 国产麻豆9l精品三级站| 久久国产一二区| 久久久久久久久岛国免费| 亚洲国产精品成人精品| 亚洲欧洲精品一区二区三区| 欧美日韩国产不卡在线看| 亚洲综合视频在线| 欧美一区二区三区男人的天堂| 伊人久久婷婷色综合98网| 欧美激情一区三区| 国产精品theporn88| 久久精品91| 欧美大片在线观看一区| 亚洲欧美日韩在线一区| 久久精品国产欧美激情| 亚洲精品人人| 在线一区欧美| 亚洲第一精品影视| 99成人精品| 精品999在线播放| 亚洲精品欧洲| 国产视频欧美视频| 亚洲国产精品久久| 国产欧美日韩视频一区二区| 欧美大片网址| 国产精品看片资源| 欧美国产欧美亚洲国产日韩mv天天看完整 | 日韩视频在线播放| 国内外成人在线| 91久久中文字幕| 国产日韩欧美一区| 亚洲精品在线电影| 尤物yw午夜国产精品视频明星| 日韩小视频在线观看专区| 国产在线欧美日韩| 一区二区三区欧美激情| 亚洲第一精品福利| 亚洲欧美日韩综合国产aⅴ| 日韩视频精品| 久久一区二区精品| 久久精品国产亚洲5555| 欧美日韩视频一区二区三区| 免费不卡中文字幕视频| 国产欧美日韩亚洲一区二区三区| 亚洲激情视频在线播放| 伊人久久亚洲热| 午夜视频在线观看一区二区| 亚洲午夜精品| 欧美日韩国产不卡| 91久久夜色精品国产九色| 影音先锋亚洲电影| 欧美一区视频| 欧美一区二区啪啪| 国产精品久久久久影院色老大| 亚洲成人中文| 在线日韩精品视频| 久久久国产精彩视频美女艺术照福利 | 国产午夜精品一区理论片飘花| 亚洲三级免费电影| 91久久黄色| 久久免费精品日本久久中文字幕| 午夜精品久久一牛影视| 欧美日韩免费视频| 最新国产成人在线观看| 亚洲国内自拍| 欧美a级大片| 亚洲国产欧美一区二区三区同亚洲| 激情亚洲网站| 久久久精品日韩| 久久亚洲精品网站| 黄色综合网站| 久久在线视频在线| 亚洲福利视频二区| aⅴ色国产欧美| 欧美日韩一区在线视频| 一区二区三区四区五区精品| 亚洲欧美国产制服动漫| 国产日产亚洲精品| 久久久www免费人成黑人精品| 免费视频一区| 亚洲老司机av| 国产精品播放| 欧美一级视频免费在线观看| 久久久99精品免费观看不卡| 在线免费日韩片| 欧美成人dvd在线视频| 亚洲免费观看高清完整版在线观看熊| 亚洲午夜精品一区二区| 国产精品久久一卡二卡| 欧美在线观看视频| 免费看黄裸体一级大秀欧美| 亚洲另类在线视频| 国产精品国产三级国产专播品爱网| 亚洲一区二区精品在线观看| 久久久999国产| 欧美另类女人| 亚洲综合社区| 欧美大香线蕉线伊人久久国产精品| 亚洲精选久久| 国产噜噜噜噜噜久久久久久久久| 欧美一区网站| 亚洲美女毛片| 麻豆久久婷婷| 亚洲一区二区三区四区五区黄 | 国产精品一卡二卡| 久久九九久精品国产免费直播| 亚洲国产精品欧美一二99| 亚洲在线第一页| 在线观看日韩| 欧美视频网站| 久久午夜电影网| 亚洲校园激情| 亚洲第一狼人社区| 久久精品噜噜噜成人av农村| 99v久久综合狠狠综合久久| 国产欧美日本在线| 欧美日韩成人综合在线一区二区| 欧美有码视频| 亚洲素人一区二区| 亚洲电影毛片| 久久人人爽人人| 亚洲欧美久久久| 99视频在线观看一区三区| 韩日欧美一区| 国产精品色网| 欧美网站大全在线观看| 免费观看成人鲁鲁鲁鲁鲁视频| 亚洲综合不卡| 一区二区高清视频在线观看| 亚洲高清网站| 免费在线观看日韩欧美| 久久成人免费网| 亚洲尤物在线| 亚洲少妇在线| 99国产精品国产精品久久| 在线精品亚洲| 国模精品一区二区三区| 国产精品一区二区你懂得| 欧美日精品一区视频| 欧美激情第五页| 欧美大片va欧美在线播放| 久久久噜噜噜久噜久久| 欧美一级二区| 欧美在线一级视频| 欧美一区二区精美| 小辣椒精品导航| 欧美一级电影久久| 欧美中文字幕在线视频| 欧美一区二区性| 久久精品国产综合精品| 久久久高清一区二区三区| 久久久久久香蕉网| 久久尤物视频| 久久精品系列| 久久综合九色欧美综合狠狠| 久久精品国产99| 久久久亚洲人| 麻豆九一精品爱看视频在线观看免费| 久久精品视频导航| 久久夜色精品国产欧美乱| 蜜桃av久久久亚洲精品| 欧美va亚洲va国产综合| 久久久久久久综合色一本| 欧美亚洲在线观看| 欧美一进一出视频| 久久久久网址| 亚洲国产三级| 亚洲精选视频免费看| 一本久道久久综合中文字幕| 在线一区二区日韩| 久久久久国产精品人| 久久九九全国免费精品观看| 国产一区二区三区精品久久久| 国产一区白浆| 91久久久久久久久| 在线视频你懂得一区|