• C++设计模式之State模式


      这里有两个例子:

          1、https://www.cnblogs.com/wanggary/archive/2011/04/21/2024117.html

      2、https://www.cnblogs.com/jiese/p/3182342.html

      关于此设计模式,有几点概括:

      Context类包含抽象的State *_state成员变量,Context类声明为State类的frend(State至少会调用Context类的ChangeState函数)。

      State是在维护一个状态,这里涉及到一个关键问题,State怎么可能不涉及到Context里的内容呢,比如,红绿灯的实际例子里,现在是红灯状态,且红灯状态有一个60秒倒计时牌。这个60秒倒计时牌的显示数字属于Context。如果我这样做:

      (先说实现的功能,进入红灯状态,显示60秒倒计时,倒计时结束,则进入红灯状态)

            1、红灯状态里响应定时器,在定时器里这样写,

                 context->CountDown = 60 - sec;         //sec初始化为0,随定时器相应增加。

                 context->UpdateWindow();                 //让倒计时牌界面显示刷新

            上面代码没什么问题,问题主要在于耦合度太大,在State里直接去修改了Context里的内容。应该怎样写,如下:

      2、红灯状态里响应定时器,在定时器里这样写,

      context->UpdateCountDownShow( count );          //UpdateCountDownShow是由Context类提供的接口

           

           总结下: 

      State只负责维护状态,而状态下的行为应由Context负责,做到状态和行为分离,降低耦合度。

      

           状态模式好像就这么简单,有点需要注意的,就是每个状态里,有维护自身状态的变量,比如上面进入红灯状态,倒计时变量应该清零,这里可以有两种做法,在ChangeState函数这个位置,1、切换状态时,下一个状态全是用new出来的对象,利用构造函数做到进入到每个状态时,状态变量“干干净净”。切换到下一个状态前,先delete掉上一个状态。2、每个状态在Context中定义好,相当于全局变量。这样在上一个状态调用context->ChangeState之前,清掉本状态相关状态变量。

      对new、delete很有信心的话用第一种方式少很多代码(把构造函数里的初始化变量提出来做为一个函数,放倒上诉第2种的情况,好像就没什么差别:))。

      2018/1/23补充:

      1、使用C++11 智能指针,上面new delete的问题都没有了。

      2、上面两个类存在交叉引用的问题,在一个类的头文件中包含另一个头文件(两个类都这样做),在某些情况下会导致编译不过(state中肯定会调用context->ChangeState()这里会编译报错,incomplte use of Context)。整理下有可以这样使用,在Context头文件里,不包含State头文件,声明State类: class State;   在State类(及子类具体状态类)中可以随意包含Context.h头文件,使用ok!

  • 相关阅读:
    java 单向链表实现
    super、this
    Java程序员们最常犯的10个错误
    Codeforces-1323D Present
    Codeforces-1323E Instant Noodles
    Codeforces-1312E Array Shrinking
    Codeforces-1327D Infinite Path
    Codeforces-1326D Prefix-Suffix Palindrome
    HDU-5885 XM Reserves
    NTT(快速数论变换)用到的各种素数及原根
  • 原文地址:https://www.cnblogs.com/kanite/p/8283627.html
Copyright © 2020-2023  润新知