實(shí)例講解C++設(shè)計(jì)模式編程中State狀態(tài)模式的運(yùn)用場(chǎng)景
State模式允許一個(gè)對(duì)象在其內(nèi)部狀態(tài)改變時(shí)改變它的行為。對(duì)象看起來(lái)似乎修改了它的類。
在面向?qū)ο笙到y(tǒng)的開(kāi)發(fā)和設(shè)計(jì)過(guò)程,經(jīng)常會(huì)遇到一種情況就是需求變更(Requirement Changing),經(jīng)常我們做好的一個(gè)設(shè)計(jì)、實(shí)現(xiàn)了一個(gè)系統(tǒng)原型,咱們的客戶又會(huì)有了新的需求。我們又因此不得不去修改已有的設(shè)計(jì),最常見(jiàn)就是解決方案就是給已經(jīng)設(shè)計(jì)、實(shí)現(xiàn)好的類添加新的方法去實(shí)現(xiàn)客戶新的需求,這樣就陷入了設(shè)計(jì)變更的夢(mèng)魘:不停地打補(bǔ)丁,其帶來(lái)的后果就是設(shè)計(jì)根本就不可能封閉、編譯永遠(yuǎn)都是整個(gè)系統(tǒng)代碼。
訪問(wèn)者模式則提供了一種解決方案:將更新(變更)封裝到一個(gè)類中(訪問(wèn)操作),并由待更改類提供一個(gè)接收接口,則可達(dá)到效果。
結(jié)構(gòu)圖:

實(shí)例:
context.h
#ifndef _CONTEXT_H_
#define _CONTEXT_H_
class State;
/**
*
**/
class Context{
public:
Context();
Context(State* state);
~Context();
void Handle();
void OperationForStateA();
void OperationForStateB();
protected:
private:
friend class State; //表明在 State 類中可以訪問(wèn)
Context 類的 private 字段,重要是訪問(wèn) ChangeState
void ChangeState(State* state);
private:
State* _state;
};
#endif //~_CONTEXT_H_
context.cpp
#include "Context.h"
#include "State.h"
#include <iostream>
using namespace std;
Context::Context(){
}
Context::Context(State* state){
this->_state = state;
}
Context::~Context(){
delete _state;
}
void Context::Handle(){
_state->Handle(this);
}
void Context::ChangeState(State* state){
///_state->ChangeState(this,state);
this->_state = state;
}
void Context::OperationForStateA(){
cout<<"Do operation in State A ";
}
void Context::OperationForStateB(){
cout<<"Do operation in State B ";
}
state.h
#ifndef _STATE_H_
#define _STATE_H_
class Context; //前置聲明
class State{
public:
State();
virtual ~State();
virtual void Handle(Context* con) = 0;
protected:
void ChangeState(Context* con,State* st);
private:
//bool ChangeState(Context* con,State* st);
};
class ConcreteStateA:public State{
public:
ConcreteStateA();
virtual ~ConcreteStateA();
void Handle(Context* con);
protected:
private:
};
class ConcreteStateB:public State{
public:
ConcreteStateB();
virtual ~ConcreteStateB();
void Handle(Context* con);
protected:
private:
};
#endif //~_STATE_H_
State.cpp
#include "State.h"
#include "Context.h"
#include <iostream>
using namespace std;
State::State(){
}
State::~State(){
}
void State::ChangeState(Context* con,State* st){
con->ChangeState(st);
}
///
ConcreteStateA::ConcreteStateA(){
}
ConcreteStateA::~ConcreteStateA(){
}
void ConcreteStateA::Handle(Context* con){
con->OperationForStateA();
cout<<":: State change from A to B"<<endl;
State::ChangeState(con,new ConcreteStateB());
}
///
ConcreteStateB::ConcreteStateB(){
}
ConcreteStateB::~ConcreteStateB(){
}
void ConcreteStateB::Handle(Context* con){
con->OperationForStateB();
cout<<":: State change from B to A"<<endl;
State::ChangeState(con,new ConcreteStateA());
}
main.cpp
#include "Context.h"
#include "State.h"
#include <iostream>
using namespace std;
int main(int argc,char* argv[]){
State* st = new ConcreteStateA();
Context* con = new Context(st);
con->Handle();
con->Handle();
con->Handle();
if (con != NULL)
delete con;
if (st != NULL)
st = NULL;
return 0;
}
可以看到在測(cè)試程序中,三次調(diào)用 con->Handle(),因?yàn)?con 狀態(tài)的不同,可以得到以下的輸出:
Do operation in State A :: State change from A to B Do operation in State B :: State change from B to A Do operation in State A :: State change from A to B
適用性
一個(gè)對(duì)象的行為取決于它的狀態(tài), 并且它必須在運(yùn)行時(shí)刻根據(jù)狀態(tài)改變它的行為。
一個(gè)操作中含有龐大的多分支的條件語(yǔ)句,且這些分支依賴于該對(duì)象的狀態(tài)。這個(gè)狀態(tài)通常用一個(gè)或多個(gè)枚舉常量表示。通常, 有多個(gè)操作包含這一相同的條件結(jié)構(gòu)。S t a t e模式將每一個(gè)條件分支放入一個(gè)獨(dú)立的類中。這使得你可以根據(jù)對(duì)象自身的情況將對(duì)象的狀態(tài)作為一個(gè)對(duì)象,這一對(duì)象可以不依賴于其他對(duì)象而獨(dú)立變化。
- 解析C++編程中如何使用設(shè)計(jì)模式中的狀態(tài)模式結(jié)構(gòu)
- 詳解C++設(shè)計(jì)模式編程中對(duì)訪問(wèn)者模式的運(yùn)用
- 深入解析C++設(shè)計(jì)模式編程中解釋器模式的運(yùn)用
- C++設(shè)計(jì)模式編程之Flyweight享元模式結(jié)構(gòu)詳解
- 詳解設(shè)計(jì)模式中的中介者模式在C++編程中的運(yùn)用
- 詳解C++設(shè)計(jì)模式編程中責(zé)任鏈模式的應(yīng)用
- 詳解設(shè)計(jì)模式中的Command命令模式及相關(guān)C++實(shí)現(xiàn)
- C++設(shè)計(jì)模式編程中使用Bridge橋接模式的完全攻略
- 設(shè)計(jì)模式中的備忘錄模式解析及相關(guān)C++實(shí)例應(yīng)用
- C++設(shè)計(jì)模式編程中簡(jiǎn)單工廠與工廠方法模式的實(shí)例對(duì)比
相關(guān)文章
基于C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易的掃雷游戲
這篇文章主要為大家詳細(xì)介紹了基于C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易的掃雷游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06
C語(yǔ)言讀取文件流的相關(guān)函數(shù)用法簡(jiǎn)介
這篇文章主要介紹了C語(yǔ)言讀取文件流的相關(guān)函數(shù)用法簡(jiǎn)介,包括fread()函數(shù)和feof()函數(shù)的使用,需要的朋友可以參考下2015-08-08
C++ STL標(biāo)準(zhǔn)庫(kù)std::vector的使用詳解
vector 是表示可以改變大小的數(shù)組的序列容器,本文主要介紹了C++ STL標(biāo)準(zhǔn)庫(kù)std::vector的使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03
C++ 中的 mutable關(guān)鍵字作用與使用場(chǎng)景分析(最新推薦)
C++中的mutable關(guān)鍵字允許在常量成員函數(shù)中修改特定成員變量,主要用于緩存機(jī)制、延遲計(jì)算和多線程同步等場(chǎng)景,它在設(shè)計(jì)中提供靈活性,但使用時(shí)需謹(jǐn)慎,本文介紹C++ 中的 mutable關(guān)鍵字作用與使用場(chǎng)景分析,感興趣的朋友一起看看吧2025-02-02
C++實(shí)現(xiàn)LeetCode(53.最大子數(shù)組)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(53.最大子數(shù)組),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07
淺析string類字符串和C風(fēng)格字符串之間的區(qū)別
string類是標(biāo)準(zhǔn)庫(kù)的類,并不是內(nèi)置類型,標(biāo)準(zhǔn)庫(kù)就像是我們自己定義的類差不多的,string類型對(duì)象沒(méi)有標(biāo)配'\0'結(jié)尾的2013-09-09
C++ 先對(duì)數(shù)組排序,在進(jìn)行折半查找
以下小編就為大家介紹兩種實(shí)現(xiàn)方法。第一種方法是,選擇排序法+循環(huán)折半查找法。第二種方法是,冒泡排序法+遞歸折半查找法。需要的朋友可以過(guò)來(lái)參考下,希望對(duì)大家有所幫助2013-10-10

