首页 > 代码库 > Effective C++ Item 35 考虑 virtual 函数以外的实现

Effective C++ Item 35 考虑 virtual 函数以外的实现

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie


1.virtual 函数版本

class GameCharacter{
public:
	virtual int healthValue() const; //返回人物的健康指数, derived classes 可重新定义它
};

2.使用 non-virtual interface 手法,那是 Template Method 设计模式的一种特殊形式。
让客户通过 public non-virtual 成员函数间接调用 private virtual 函数
class GameCharacter{
public:
	int healthValue() const // Item 36 derived classes不重新定义它
	{
		//...
		int retVal = doHealthValue();
		//...
		return retVal;
	}
private:
	virtual int doHealthValue() const //derive classes 可重新定义它
	{
		//...  缺省算法
	}
};


3.将 virtual 函数替换为“函数指针成员变量”,这是 Strategy 设计模式的一种分解表现形式。
优点:每天个对象可各自自己的健康计算函数、可在运行期改变计算函数
缺点:可能必须降低 GameCharacter 封装性。例如 声明 non-member 函数为 friends 或为其实现的某一部分提供 public 访问函数
//1.藉由 Function Pointers 实现 Strategy 模式


int defaultHealthCalc(const GameCharacter &gc);<span style="font-family: Arial, Helvetica, sans-serif;">//计算健康指数的缺省算法</span>
class GameCharacter{
public:
	typedef int (*HealthCalcFunc)(const GameCharacter);
	explicit GameCharacter(HealthCalcFunc hcf = defaultHealthCalc) : healthFunc(hcf){}
	int healthValue() const{
		return healthFunc(*this);
	}
	//...
private:
	HealthCalcFunc healthFunc;
};

//2.藉由 tr1::function 完成 Strategy 模式
//以 tr1::function 成员变量替换 virtual 函数,因而允许使用任何可调用物(callable entity)
搭配一个兼容于需求的签名式。这也是 Strategy 设计模式的某种形式。

class GameCharacter;
int defaultHealthCalc(const GameCharacter &gc);
class GameCharacter{
public:
	//HealthCalcFunc 可以是任何“可调用物”,可被调用并被接受任何兼容于 GameCharacter 之物,返回任何兼容于 int的东西
	typedef std::tr1::function<int (const GameCharacter&)> HealthCalcFunc;
	explicit GameCharacter(HealthCalcFunc hcf = defaultHealthCalc) : healthFunc(hcf){}
	int healthValue() const{
		return healthFunc(*this);
	}
	//...
private:
	HealthCalcFunc healthFunc;
};


4.将继承体系内的 virtual 函数替换为另一个继承体系内的 virtual 函数。这是 Strategy 设计模式
的传统实现手法。
class GameCharacter;
class HealthCalcFunc{
public:
	//...
	virtual int calc(const GameCharacter &gc) const{...}
};


HealthCalcFunc defaultHealthCalc;
class GameCharacter{
public:
	explicit GameCharacter(HealthCalcFunc *phcf = &defaultHealthCalc) : pHealthCalc(phhcf){}
	int healthValue() const {return pHealthCalc->calc(*this);}
private:
	HealthCalcFunc *pHealthCalc;
};