博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
C++ 之 策略模式
阅读量:4678 次
发布时间:2019-06-09

本文共 4021 字,大约阅读时间需要 13 分钟。

1  会飞的鸭子 

  Duck 基类,含成员函数 Swim() 和 Display();派生类 MallardDuck,RedheadDuck 和 RubberDuck,各自重写 Display()

class Duck {public:    void Swim();    virtual void Display();};class MallardDuck : public Duck {public:    void Display(); // adding virtual is OK but not necessary};class RedheadDuck ... class RubberDuck ...

  现在要求,为鸭子增加飞的技能 -- Fly,应该如何设计呢?

1.1  继承

  考虑到并非所有的鸭子都会飞,可在 Duck 中加普通虚函数 Fly(),则“会飞”的继承 Fly() ,“不会飞”的重写 Fly()

void Duck::Fly() {  std::cout << "I am flying !" << std::endl;  }void RubberDuck::Fly() {  std::cout << "I cannot fly !" << std::endl;  }

1.2  接口

  用普通虚函数并非良策, “1.2 普通虚函数” 中已经解释。代替方法是 “纯虚函数 + 缺省实现”,即将基类中的 Fly() 声明为纯虚函数,同时写一个缺省实现

  因为是纯虚函数,所以只有“接口”会被继承,而缺省的“实现”却不会被继承,是否调用 Fly() 的缺省实现,则取决于重写的 Fly()

void MallardDuck::Fly() { Duck::Fly(); } void RedheadDuck::Fly() { Duck::Fly(); }

1.3  设计模式

  到目前为止,并没有设计模式,但问题已经解决了。实际上用不用设计模式,取决于实际需求,也取决于开发者。

  <Design Patterns> 中,关于策略模式的适用情景,如下所示:

1) many related classes differ only in their behavior

2) you need different variants of an algorithm

3) an algorithm uses data that clients shouldn't know about

4) a class defines many behaviors, and these appear as multiple conditional statements in its operations

  显然,鸭子的各个派生类属于 “related classes”。关键就在于“飞”这个行为,如果只是将“飞”的行为,简单划分为“会飞”和“不会飞”,则不用设计模式完全可以。

  如果“飞行方式”,随着派生类的增多,至少会有几十种;或者视“飞行方式”为一种算法,以后还会不断改进;再或“飞行方式”作为封装算法,提供给第三方使用那么此时,设计模式的价值就体现出来了 -- 易复用,易扩展,易维护

  而第 4) 种适用情景,多见于重构之中,取代一些条件选择语句 -- "Replace Type Code with State/Strategy"

 

2  设计原则

  在引出策略模式之前,先看面向对象的三个设计原则

1)  隔离变化identify what varies and separate them from what stays the same

   Duck 基类中, “飞行方式“是变化的,于是把 Fly() 择出来,和剩余不变的分隔开来

2)  编程到接口program to an interface, not an implementation

  分离Fly(),将其封装为一个接口,里面实现各种不同的“飞行方式” (一系列”算法“),添加或修改算法都在这个接口里进行。

  “接口”对应于 C++ 便是抽象基类,故可将“飞行方式”封装为 FlyBehavior 类,并在类中声明 Fly() 为纯虚函数

class FlyBehavior {public:    virtual void Fly() = 0;};class FlyWithWings : public FlyBehavior {public:    virtual void Fly();};class FlyNoWay ... class FlyWithRocket ...

  具体实现各种不同的算法 -- “飞行方式”,如下:

void FlyWithWings::Fly() {  std::cout << "I am flying !" << std::endl;  }void FlyNoWay::Fly() {  std::cout << "I cannot fly !" << std::endl;  }void FlyWithRocket::Fly() {  std::cout << "I am flying with a rocket !" << std::endl; }

3)  复合 > 继承:favor composition (has-a) over inheritance (is-a)

   公有继承即是 “is-a”,而 Composition (复合或组合) 的含义是 “has-a”,因此,可在 Duck 基类中,声明 FlyBehavior 型指针,如此,只需通过指针 _pfB 便可调用相应的”算法“ -- ”飞行方式“

class Duck {
...private: FlyBehavior* fb_; // 或 std::unique_ptr
fb_;};

 

3  策略模式

3.1  内容

  即便不懂设计模式,只要严格按照遵守 隔离变化 --> 编程到接口 --> 复合 三个原则,则设计思路也会和策略模式类似:

  下面是策略模式的具体内容:

  Defines a family of algorithms,  encapsulates each one,  and makes them interchangeable.  Strategy lets the algorithm vary independently from clients that use it.

 

  Context 指向 Strategy (由指针实现);Context 通过 Strategy 接口,调用一系列算法;ConcreteStrategy 实现了一系列具体的算法

3.2  智能指针

  上例中,策略模式的“接口” 对应于 FlyBehavior 类,“算法实现”分别对应派生类 FlyWithWings, FlyNoWay, FlyWithRocket,“引用”对应 fb_ 指针

  为了简化内存管理,可将 fb_ 声明为一个“智能指针”,如此,则不需要手动实现析构函数,采用编译器默认生成的即可。

Duck::Duck(FlyBehavior *fb)    : fb_(fb){}

3.3  分析 

  直观上看, Duck 对应于 Context,实际上是其派生类 MallardDuck 等,通过 FlyBehavior 接口来调用各种“飞行方式”。因此,需要在各个派生类的构造函数中,初始化 fb_

MallardDuck::MallardDuck(FlyBehavior *fb)    : Duck(fb){}

 然后,在 Duck 基类中,通过指针 fb_, 实现对 Fly() 的调用

void Duck::PerformFly(){    fb_->Fly();}

  除了在构造函数中初始化 fb_ 外,还可在 Duck 类中,定义一个 SetFlyBehavior 成员函数,动态的设置“飞行方式”

void Duck::SetFlyBehavior(FlyBehavior *fb){
fb_ = fb;}

3.4  main 函数  

  因为 main 执行结束后,程序也就结束了,所以对于简单程序,new 了指针后,可以不用 delete

int main (){    FlyBehavior *pfWings = new FlyWithWings;    FlyBehavior *pfNo = new FlyNoWay;    FlyBehavior *pfRocket = new FlyWithRocket;    // fly with wings    Duck *pDuck = new MallardDuck(pfWings);    pDuck->PerformFly();    // fly with a rocket    pDuck->SetFlyBehavior(pfRocket);    pDuck->PerformFly();}

  

 代码链接:

 

小结

1)  面向对象的三个设计原则:隔离变化,编程到接口,复合 > 继承

2)  策略模式主要涉及的是“一系列算法“,熟悉其适用的四种情景

 

参考资料

 <大话设计模式> 第二章

 <Head First Design Patterns> chapter 1

 <Effective C++> item 32, item 38

 <Design Patterns> Strategy

 <Refactoring> chapter 8

 

转载于:https://www.cnblogs.com/xinxue/p/5271184.html

你可能感兴趣的文章
常用的匹配正则表达式和实例
查看>>
小组成员及其git链接
查看>>
SQL case when else
查看>>
JAVA学习之路(环境配置,)
查看>>
Task.WaitAll代替WaitHandle.WaitAll
查看>>
MVc Identity登陆锁定
查看>>
cdn连接失败是什么意思_关于CDN的原理、术语和应用场景那些事
查看>>
ultraedit26 运行的是试用模式_免费试用U盘数据恢复工具 – 轻松找回U盘丢失的各种数据!...
查看>>
怎么从转移特性曲线上看dibl_白话IVD中的流体——泵的流量特性与管路阻力特性...
查看>>
奈奎斯特与香农定理_通俗理解奈奎斯特带宽
查看>>
ercharts一个页面能放几个_谷歌优化排名网站内页,一般放置几个关键词?
查看>>
redirect路由配置 vue_Vue 动态生成路由结构
查看>>
maven仲裁机制_Maven 基础知识依赖机制
查看>>
canvas绘制四分之一圆_用canvas画太极图(一步步详解附带源代码)
查看>>
计算上个月的第一天和最后一天_20年的最后一场旅行,21年的第一场旅行
查看>>
抄表 软件_水表远程抄表方案 M-BUS NB-IOT LoRa有什么区别呢
查看>>
一般柱子与柱子的距离_建筑内部布置柱子 间距大概是多少?
查看>>
python比excel好在哪_在数据分析方面,比起python,excel的局限性在哪(python excle 图表)...
查看>>
python 语言爱好者_语言都是相通的,学好一门语言,再学第二门语言就很简单,记录一下我复习c语言的过程。...
查看>>
input层级高 小程序_关于微信小程序textarea层级过高问题解决
查看>>