思考题

  1. public class GumballMachine {
  2. final static int SOLD_OUT = 0;
  3. final static int NO_QUARTER = 1;
  4. final static int HAS_QUARTER = 2;
  5. final static SOLD = 3;
  6. int state = SOLD_OUT;
  7. int count = 0;
  8. public GumballMachine(int count) {
  9. this.count = count;
  10. if(count > 0) {
  11. state = NO_QUARTER;
  12. }
  13. }
  14. public void insertQuarter() {
  15. if(state == HAS_QUARTER) {
  16. // print error message
  17. } else if(state == NO_QUARTER) {
  18. state = HAS_QUARTER;
  19. // print success message
  20. } else if(state == SOLD_OUT) {
  21. // print error message
  22. } else if(state == SOLD) {
  23. // print error message
  24. }
  25. }
  26. public void ejectQuarter() {
  27. // ...
  28. }
  29. public void turnCrank() {
  30. // ...
  31. }
  32. public void dispense() {
  33. // ...
  34. }
  35. }

下列哪一项描述了我们实现的状态?(多选) P396

  • [x] A. 这份代码确实没有遵守开放-关闭原则

    • 当新增状态时,必须在所有方法中加上对新状态的条件判断,所以没有遵守开放-关闭原则
  • [ ] B. 这份代码会让 Fortran 程序员感到骄傲
    • 不知道为什么
    • 【答案有此选项】
  • [x] C. 这个设计其实不符合面向对象
    • 这个设计是面向过程的,所有的操作都通过条件判断,没有封装状态
  • [x] D. 状态转换被埋藏在条件语句中,所以并不明显
    • 状态转换是在行为方法内的条件语句中,要找到状态转换前后的状态需要阅读行为方法内的全部代码,难以快速了解某种状态会如何转换
  • [x] E. 我们还没有把会改变的那部分包起来
    • 状态和行为都会改变,但行为比较固定且与实际相对应,状态是抽象出来的,所以应该将状态封装起来
  • [x] F. 未来加入的代码很有可能会导致 bug
    • 由于所有行为方法内都有不同状态的条件判断,所以在任何状态发生变化时,都要对所有行为方法进行修改进行处理,很容易遗忘对某行为方法的修改

思考题

  1. public class GumballMachine {
  2. State soldOutState;
  3. State noQuarterState;
  4. State hasQuarterState;
  5. State soldState;
  6. State state = soldOutState;
  7. int count = 0;
  8. public GumballMachine(int numberGumballs) {
  9. soldOutState = new SoldOutState(this);
  10. noQuarterState = new NoQuarterState(this);
  11. hasQuarterState = new HasQuarterState(this);
  12. soldState = new SoldState(this);
  13. this.count = numberGumballs;
  14. if (numberGumballs > 0) {
  15. state = noQuarterState;
  16. }
  17. }
  18. public void insertQuarter() {
  19. state.insertQuarter();
  20. }
  21. public void ejectQuarter() {
  22. state.ejectQuarter();
  23. }
  24. public void turnCrank() {
  25. state.turnCrank();
  26. state.dispense();
  27. }
  28. void setState(State state) {
  29. this.state = state;
  30. }
  31. void releaseBall() {
  32. // print success message
  33. if(count != 0) {
  34. count = count - 1;
  35. }
  36. }
  37. }

让我们来回头看看糖果机的实现。如果曲柄被转动了,但是没有成功(比方说顾客没有先投入25分钱的硬币)。在这种情况下,尽管没有必要,但我们还是会调用 dispense() 方法。对于这个问题你要如何修改呢? P405

  • State 接口的 turnCrank() 方法增加返回值以表示是否正确处理,只有在正确处理时,才调用 dispense() 方法

状态模式

允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它的类。 P410

特点

  • 将每个状态行为局部化到自己的状态类中 P407
  • 让每个状态“对修改更换比”,让上下文“对扩展开放”,因为可以加入新的状态类 P407

缺点

  • 通常会导致设计中类的数目大量增加 P423

状态模式和策略模式的区别

状态模式

  • 将一群行为封装在状态对象中,上下文的行为随时可委托到那些状态对象中的一个。当前状态会在状态对象集合中游走改变,以反应出上下文内部的状态,因此,上下文的行为也会跟着改变。但时客户对于上下文的状态对象了解不多,甚至根本是浑然不知 P411
  • 是不用在上下文中放置许多条件判断的替代方案,通过将行为包装进状态对象中,在上下文内简单地改变状态对象来改变上下文的行为 P411

策略模式

  • 客户通常主动指定上下文所要组合的策略对象 P411
  • 是除继承之外的一种弹性替代方案,可以通过组合不同的对象来改变行为 P411

思考题

应该由状态类还是上下文决定状态转换的流向? P412

  • 当状态转换是固定的时候,适合放在上下文中(此时状态类之间不相互依赖,是对状态类修改封闭) P412
  • 当状态转换是更动态的时候,通常就会放在状态类中(此时状态类之间产生了依赖,是对上下文修改封闭) P412

思考题

我们需要你为糖果机写一个重填糖果的 refill() 方法。这个方法需要一个变量——所要填入机器中的糖果数目。它应该能更新糖果机内的糖果数目,并重设机器的状态。 P421

  1. void refill(int num) {
  2. this.count += num;
  3. if(state instanceof SoldOutState) {
  4. state = noQuarterState;
  5. }
  6. }

思考题

配对下列模式和描述: P422

状态模式:封装基于状态的行为,并将行为委托到当前状态

策略模式:将可以互换的行为封装起来,然后使用委托的方法,决定使用哪一个行为

模板方法模式:由子类决定如何实现算法中的某些步骤

本文首发于公众号:满赋诸机(点击查看原文) 开源在 GitHub :reading-notes/head-first-design-patterns

Head First 设计模式 —— 12. 状态 (State) 模式的更多相关文章

  1. Java 实现状态(State)模式

    /** * @author stone */ public class WindowState { private String stateValue; public WindowState(Stri ...

  2. 状态(State)模式--设计模式

    定义与特点 1.1 定义 状态模式允许一个对象在其内部状态改变的时候改变其行为.这个对象看上去就像是改变了它的类一样. 1.2 特点 状态模式优点: 封装了转换规则,并枚举可能的状态,它将所有与某个状 ...

  3. [设计模式][c++]状态切换模式

    转自:http://blog.csdn.net/yongh701/article/details/49154439 状态模式也是设计模式的一种,这种设计模式思想不复杂,就是实现起来的代码有点复杂.主要 ...

  4. 状态(State)模式

    状态模式,又称状态对象模式(Pattern of Objects for States),状态模式是对象的行为模式.状态模式允许一个对象在其内部状态改变的时候改变其行为.这个对象看上去就像是改变了它的 ...

  5. 设计模式C++描述----16.状态(State)模式

    一. 举例 一般汽车发动机工作时有四种状态,吸气.压缩.做功和排气. 在运行时,不同的状态会有不同的行为,当前的状态机在适当的时候会过渡到下一状态. 其实用户在使用时根本不知道当前的状态,也无需知道当 ...

  6. C#设计模式(12)——享元模式(Flyweight Pattern)

    一.引言 在软件开发过程,如果我们需要重复使用某个对象的时候,如果我们重复地使用new创建这个对象的话,这样我们在内存就需要多次地去申请内存空间了,这样可能会出现内存使用越来越多的情况,这样的问题是非 ...

  7. 设计模式--模板方法 And State模式

    1.模板方法   钩子: 在抽象基类已经有默认的定义,子类选择是否覆盖它 在模板方法模式中,  抽象基类中使用 钩子函数(子类视情况是否覆盖)  来达到控制模板方法中  流程控制的 目的 设计原则: ...

  8. C#设计模式(12)——享元模式(Flyweight Pattern)(转)

    一.引言 在软件开发过程,如果我们需要重复使用某个对象的时候,如果我们重复地使用new创建这个对象的话,这样我们在内存就需要多次地去申请内存空间了,这样可能会出现内存使用越来越多的情况,这样的问题是非 ...

  9. java设计模式-----12、外观模式

    Facade模式也叫外观模式,是由GoF提出的23种设计模式中的一种.Facade模式为一组具有类似功能的类群,比如类库,子系统等等,提供一个一致的简单的界面.这个一致的简单的界面被称作facade. ...

随机推荐

  1. TCP连接时动态端口的相关问题说明

    最近在线上遇到一个TCP动态端口相关的问题,之前没有留意过此类问题,做个笔记记录在这里,希望也能给大家提供个参考. 简单介绍下问题的场景:Windows服务器上,部署了网关程序SG和RPC进程,其中R ...

  2. k8s遇见的问题

    open /etc/docker/certs.d/registry.access.redhat.com/redhat-ca.crt: no such file or directory 解决方案   ...

  3. React Native学习记录

    1.端口问题 在调试的时候,监听的是8081端口.如果被占用,会报错,并且在reload的时候导致app直接崩掉. 2.插件网站收集 https://nativebase.io/ https://js ...

  4. 【ubantu 安装Jmeter和Jdk环境】

    Linux环境安装Java(含安装包下载地址) 一定要使用有权限的用户 1.下载JDK压缩包,下载地址:https://blog.csdn.net/duketyson2009/article/deta ...

  5. JS怎么把for循环出来的东西放到一个数组里

    var students=[ {name: "vehicleTravelLicenseCopyBack", id: "a1"}, {name: "ve ...

  6. Rest vs Soap 比较

    一.前言 昨天为止,政府的一个公共部门的项目顺利结束,就系统间消息传输这个点,知识点总结一下.本文主要参考octoperf的文章,链接见文末参考资料. 系统中用到的是SOAP协议进行传输数据,有人会立 ...

  7. Docker(六):Docker安装Kibana

    查找Kibana镜像 镜像仓库 https://hub.docker.com/ 下拉镜像 docker pull kibana:7.7.0 查看镜像 docker images 创建Kibana容器 ...

  8. JVM虚拟机(二):字节码执行引擎

    运行时栈帧结构     栈帧是用于支持虚拟机进行方法调用和方法执行背后的数据结构,它也是虚拟机运行时数据区中的虚拟机栈的栈元素.栈帧存储了方法的局部变量表.操作数栈.动态链接.和方法返回地址等信息. ...

  9. Kubernetes弃用Docker后怎么办?

    本文转自Rancher Labs 近期,Kubernetes在其最新的Changelog中宣布,自Kubernetes 1.20之后将弃用Docker作为容器运行时.这一消息在云原生领域激起了不小的水 ...

  10. remmina 软件rdp协议链接windows失败

    remmina 1.42  链接 win10 提示失败......其他版本win还没有测试过. 忘记了在那个论坛有是说加密问题,照着改确实可以.具体原因是默认设置加密方式这一项不知道为什么不起作用,手 ...