c++ 设计模式4 (Strategy)
2016-02-19 10:22
651 查看
3.2 Strategy (策略模式)
动机:在软件构建过程中,某些对象使用的算法可能多种多样,经常改变,如果将这些算法都编码到对象中,将会使对象变得异常复杂,而且有时候支持不使用或不常使用的算法也是一个性能负担。
如何在运行时根据需要透明地改变对象的算法?将算法和对象本身解耦,从而避免上述问题?
代码示例:
如果使用第一种方法,需要更改枚举类型和if else语句(第6行和24行),违背了开放封闭原则(对扩展开发,对修改封闭);
采用第二种方法(策略模式),定义税法基类,对不同国家税法定义不同子类,override Calculate方法。59行,多态调用。
假设出现变化:扩展32 - 39行,SalesOrder不变(复用性,遵循开放封闭原则)。
补:面向对象提到的复用性,指的是编译单位(二进制单位)的复用性,不是粘贴代码的复用性
定义: 定义一系列算法, 把它们一个个封装起来,并且使它们可以互相替换(变化)。该模式使得算法可独立于使用它的客户程序(稳定)而变化(扩展,子类化)。
类图:
要点总结:
Strategy及其子类为组件提供了一系列可重用的算法,从而可以得到类型在运行时方便地根据需要在各个算法之间进行切换。
Strtegy模式提供了用条件判断语句以外的另一种选择消除条件判断语句,就是在解耦合。含有许多条件判断语句的代码通常都需要Strategy模式。(除非if else语句绝对不变,如对一周七天判断等)
如果Strategy对象没有实例变量,那么各个上下午可以共享同一个Strategy对象,从而节省对象开销。
动机:在软件构建过程中,某些对象使用的算法可能多种多样,经常改变,如果将这些算法都编码到对象中,将会使对象变得异常复杂,而且有时候支持不使用或不常使用的算法也是一个性能负担。
如何在运行时根据需要透明地改变对象的算法?将算法和对象本身解耦,从而避免上述问题?
代码示例:
如果使用第一种方法,需要更改枚举类型和if else语句(第6行和24行),违背了开放封闭原则(对扩展开发,对修改封闭);
采用第二种方法(策略模式),定义税法基类,对不同国家税法定义不同子类,override Calculate方法。59行,多态调用。
假设出现变化:扩展32 - 39行,SalesOrder不变(复用性,遵循开放封闭原则)。
补:面向对象提到的复用性,指的是编译单位(二进制单位)的复用性,不是粘贴代码的复用性
// strategy1.cpp enum TaxBase { CN_Tax, US_Tax, DE_Tax, FR_Tax //更改 }; class SalesOrder{ TaxBase tax; public: double CalculateTax(){ //... if (tax == CN_Tax){ //CN*********** } else if (tax == US_Tax){ //US*********** } else if (tax == DE_Tax){ //DE*********** } else if (tax == FR_Tax){ //更改 //... } //.... }
//Strategy.cpp class TaxStrategy{ public: virtual double Calculate(const Context& context)=0; virtual ~TaxStrategy(){} }; class CNTax : public TaxStrategy{ public: virtual double Calculate(const Context& context){ //*********** } }; class USTax : public TaxStrategy{ public: virtual double Calculate(const Context& context){ //*********** } }; class DETax : public TaxStrategy{ public: virtual double Calculate(const Context& context){ //*********** } }; //扩展 //********************************* class FRTax : public TaxStrategy{ public: virtual double Calculate(const Context& context){ //......... } }; class SalesOrder{ private: TaxStrategy* strategy; public: SalesOrder(StrategyFactory* strategyFactory){ this->strategy = strategyFactory->NewStrategy(); } ~SalesOrder(){ delete this->strategy; } public double CalculateTax(){ //... Context context(); double val = strategy->Calculate(context); //多态调用 //... } };
定义: 定义一系列算法, 把它们一个个封装起来,并且使它们可以互相替换(变化)。该模式使得算法可独立于使用它的客户程序(稳定)而变化(扩展,子类化)。
类图:
要点总结:
Strategy及其子类为组件提供了一系列可重用的算法,从而可以得到类型在运行时方便地根据需要在各个算法之间进行切换。
Strtegy模式提供了用条件判断语句以外的另一种选择消除条件判断语句,就是在解耦合。含有许多条件判断语句的代码通常都需要Strategy模式。(除非if else语句绝对不变,如对一周七天判断等)
如果Strategy对象没有实例变量,那么各个上下午可以共享同一个Strategy对象,从而节省对象开销。
相关文章推荐
- C与C++中使用带默认值的参数
- Typical memory leak (C++中典型的内存泄露)
- 纯C语言实现简单继承机制
- 转:C语言面试题大汇总 (图像处理方向)
- ❤️C++的继承
- ❤️C++基础语法
- ❤️C++应用场景
- ❤️C++语言特点:
- utilities——C++常用仿函数
- C++模板学习
- C++ vector
- 判断素数
- C语言实现单链表逆序与逆序输出实例
- 求1+2+3+4+....+100
- 数字按照大小排列
- C++编译器生成的构造函数的总结
- 春节后第一波大优惠来袭!!史上最全最强C语言视频课程(全程字幕 + 习题作业)
- 在C++中如何使用msgpack进行对象的序列化
- 我应该记录一下我不太了解的一些c语言函数
- 几个数中比较找最大的