前面模板方法模式是“堅(jiān)持相同的代碼“,而被覆蓋的函數(shù)是“變化的代碼“。然而,這種變化在編譯時(shí)通過繼承被固定下來。按照“組合優(yōu)于繼承“的格言,可以利用組合來解決將變化的代碼從“堅(jiān)持相同的代碼“中分開的問題,從而產(chǎn)生策略(Strategy)模式。這種方法的一個(gè)明顯的好處:在程序運(yùn)行時(shí),可以插入變化的代碼。策略模式也加入“語境“,它可以是一個(gè)代理類,這個(gè)類控制著對(duì)特定策略對(duì)象的選擇和使用。
“策略“的意思就是:可以使用多種方法來解決某個(gè)問題,即條條大陸通羅馬。現(xiàn)在考慮一下忘記了某個(gè)人姓名時(shí)的情景。這里的程序可以用不同方法解決這個(gè)問題,實(shí)例代碼如下:
#include<iostream>
using namespace std;
class NameStrategy
{
public:
virtual void greet()=0;
};
class SayHi: public NameStrategy
{
public:
void greet()
{
cout<<"Hi! How's it going?"<<endl;
}
};
class Ignore: public NameStrategy
{
public:
void greet()
{
cout<<"Pretend I don't see you)"<<endl;
}
};
class Admission:public NameStrategy
{
public:
void greet()
{
cout<<"I'm sorry ,I forgot your name."<<endl;
}
};
class Context
{
NameStrategy & strategy;
public:
Context(NameStrategy & strat):strategy(strat){}
void greet(){strategy.greet();}
};
int main()
{
SayHi sayhi;
Ignore ignore;
Admission admission;
Context c1(sayhi),c2(ignore),c3(admission);
c1.greet();
c2.greet();
c3.greet();
}
Context::greet()可以正規(guī)地寫得更加復(fù)雜些,它類似模板方法模式,因?yàn)槠渲邪瞬荒芨淖兊拇a。但在函數(shù)main()中可以看到,可以在運(yùn)行時(shí)就策略進(jìn)行選擇。更進(jìn)一步的做法。可以將狀態(tài)模式與Context對(duì)象的生存期間變化的策略模式結(jié)合起來使用。