设计模式学习专栏十一--------状态模式

设计模式学习专栏十一--------状态模式

名称: 状态模式 (State)

价值观念: 通过改变对象内部的状态来帮助对象控制自己的行为

场景


设计一个万能糖果机 , 我们希望设计尽可能有弹性 , 而且将来我们可能要为它增加更多的行为~

image

刚开始的设计方式

public class GumballMachine {
 
    final static int SOLD_OUT = 0;
    final static int NO_QUARTER = 1;
    final static int HAS_QUARTER = 2;
    final static int SOLD = 3;
 
    int state = SOLD_OUT;
    int count = 0;
  
    public GumballMachine(int count) {
        this.count = count;
        if (count > 0) {
            state = NO_QUARTER;
        }
    }
  
    public void insertQuarter() {
        if (state == HAS_QUARTER) {
            System.out.println("You can't insert another quarter");
        } else if (state == NO_QUARTER) {
            state = HAS_QUARTER;
            System.out.println("You inserted a quarter");
        } else if (state == SOLD_OUT) {
            System.out.println("You can't insert a quarter, the machine is sold out");
        } else if (state == SOLD) {
            System.out.println("Please wait, we're already giving you a gumball");
        }
    }
    
    public void ejectQuarter() {
        if (state == HAS_QUARTER) {
            System.out.println("Quarter returned");
            state = NO_QUARTER;
        } else if (state == NO_QUARTER) {
            System.out.println("You haven't inserted a quarter");
        } else if (state == SOLD) {
            System.out.println("Sorry, you already turned the crank");
        } else if (state == SOLD_OUT) {
            System.out.println("You can't eject, you haven't inserted a quarter yet");
        }
    }
    ......
}

我们的第一版设计完成了 , 发现每个动作下都需要判断当前的状态,然后做出相应的动作.

新功能引入 , 我们加入了一个新的游戏状态 , 当曲柄被转动时,有10%的几率掉下来的是两颗糖果(赢家状态)

此时我们会发现第一版设计中. 每个动作下 都需要新增条件判断 , 违反了对修改关闭的原则. 程序出错概率大大提升.

如何解决


分析程序扩展时的 可变部分与不变部分

insertCoin returnCoin trunCrank dispense
OnReadyState - - - -
HasCoin - - - -
SoldState - - - -
SoldOutState - - - -
WinnerState *** *** *** ***

不变部分: 从横向来看。 用户能执行的操作都是一样的。 (插入硬币,按下退币按钮,拉下把手)

变化部分: 从横向看。如果糖果工厂新增的状态, 对于用户每一种动作,糖果机的响应都是不同的。都要做出对应的修改

将可变部分抽取出来: 每一种状态都会执行4种操作,糖果机具体的操作与当前状态有关。 因此将状态与该状态下的对应的动作行为抽取出来形成接口。让每一个状态都实现该接口。

image

状态模式总览


定义:允许对象在内部状态在改变时改变它的行为,对象看起来好像改变了它的类
(将状态与该状态下的行为封装成独立的类,并将动作委托到代表当前状态的对象)

  • 模式的理解

    • 类图

      image
    • 角色

      • 上下文Context
      • 封装状态及该状态下行为的 State
      • 具体的状态实习那类ConcreteState
    • 细节

      • 状态模式允许一个对象基于内部状态而拥有不同的行为
      • 通过把每个状态封装进一个类, 我们把以后需要做的任何改变都局部化了 (改变这个状态下的行为)
      • 状态模式与策略模式有相同的类图 ,但是它们的意图不同
        • 策略模式通常会用行为或算法来配置Context类
        • 状态模式允许Context随着状态的改变而改变行为
      • 状态转换可以由State类(某个行为后改变状态)或者Context(外部设置状态setState)类控制.
      • 使用状态模式通常会导致设计中的类的数据大量增量(状态类)

核心代码部分

  • 上下文

    public class GumballMachine {
     
        //上下文持有不同的状态引用
      State soldOutState;     
      State noQuarterState;
      State hasQuarterState;
      State soldState;
      State winnerState;
     
      State state = soldOutState;
      int count = 0;
     
      public GumballMachine(int numberGumballs) {
          soldOutState = new SoldOutState(this);
          noQuarterState = new NoQuarterState(this);
          hasQuarterState = new HasQuarterState(this);
          soldState = new SoldState(this);
          winnerState = new WinnerState(this);
    
          this.count = numberGumballs;
          if (numberGumballs > 0) {
              state = noQuarterState;
          } 
      }
        //上下文提供改变状态的接口
        void setState(State state) {
          this.state = state;
      }
     
      public void insertQuarter() {
          state.insertQuarter();  //将行为委托给状态对象来处理
      }
     
      public void ejectQuarter() {
          state.ejectQuarter();
      }
     
      public void turnCrank() {
          state.turnCrank();
          state.dispense();
      }
     
      void releaseBall() {
          System.out.println("A gumball comes rolling out the slot...");
          if (count != 0) {
              count = count - 1;
          }
      }
     
      int getCount() {
          return count;
      }
     
      void refill(int count) {
          this.count += count;
          System.out.println("The gumball machine was just refilled; it's new count is: " + this.count);
          state.refill();
      }
    
        getter and setter...
    }
    
  • 状态接口

    public interface State {
     
      public void insertQuarter();
      public void ejectQuarter();
      public void turnCrank();
      public void dispense();
      
      public void refill();
    }
    
  • 具体的状态
public class WinnerState implements State {
    GumballMachine gumballMachine;
 
    public WinnerState(GumballMachine gumballMachine) {
        this.gumballMachine = gumballMachine;
    }
 
    public void insertQuarter() {
        System.out.println("Please wait, we're already giving you a Gumball");
    }
 
    public void ejectQuarter() {
        System.out.println("Please wait, we're already giving you a Gumball");
    }
 
    public void turnCrank() {
        System.out.println("Turning again doesn't get you another gumball!");
    }
 
    public void dispense() {
        gumballMachine.releaseBall();
        if (gumballMachine.getCount() == 0) {
            //通过上下文改变状态
            gumballMachine.setState(gumballMachine.getSoldOutState());
        } else {
            gumballMachine.releaseBall();
            System.out.println("YOU'RE A WINNER! You got two gumballs for your quarter");
            if (gumballMachine.getCount() > 0) {
                gumballMachine.setState(gumballMachine.getNoQuarterState());
            } else {
                System.out.println("Oops, out of gumballs!");
                gumballMachine.setState(gumballMachine.getSoldOutState());
            }
        }
    }
 
    public void refill() { }
    
    public String toString() {
        return "despensing two gumballs for your quarter, because YOU'RE A WINNER!";
    }
}
  • 主程序

    public class GumballMachineTestDrive {
    
      public static void main(String[] args) {
          GumballMachine gumballMachine =
              new GumballMachine(10);
    
          System.out.println(gumballMachine);
    
          gumballMachine.insertQuarter();
          gumballMachine.turnCrank();
          gumballMachine.insertQuarter();
          gumballMachine.turnCrank();
    
          System.out.println(gumballMachine);
      }
    }
    
  • 输出结果

    Mighty Gumball, Inc.
    Java-enabled Standing Gumball Model #2004
    Inventory: 10 gumballs
    Machine is waiting for quarter
    
    You inserted a quarter
    You turned...
    A gumball comes rolling out the slot...
    You inserted a quarter
    You turned...
    A gumball comes rolling out the slot...
    
    Mighty Gumball, Inc.
    Java-enabled Standing Gumball Model #2004
    Inventory: 8 gumballs
    Machine is waiting for quarter
    

参考

​ 书籍: HeadFirst设计模式

​ 代码参考地址: 我就是那个地址

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容

  • 高阶特性 支持垃圾回收对运行时的一个深远影响是所有代码都需要做额外的记录。而类型安全也有一个重要影响,即要求对程序...
    懿民阅读 952评论 0 0
  • 寒夜[宋]杜耒 寒夜客来茶当酒,竹炉汤沸火初红。 寻常一样窗前月,才有梅花便不通。 在小寒的夜晚,杜耒非常的孤独,...
    BestHenry阅读 358评论 0 1