1、模式简介

状态模式的定义:

  状态模式允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它的类。这个模式将状态封装成独立的类,并将动作委托到代表当前状态的类的对象。

状态模式的优点:

  • 封装了转换规则;
  • 枚举可能的状态,在枚举状态之前需要确定状态种类;
  • 将所有与某个状态有关的行为放到一个类中,并且可以方便地增加新的状态,只需要改变对象状态即可改变对象的行为;
  • 允许状态转换逻辑与状态对象合成一体,而不是某一个巨大的条件语句块;
  • 可以让多个环境对象共享一个状态对象,从而减少系统中对象的个数。

状态模式的缺点:

  • 状态模式的使用必然会增加系统类和对象的个数;
  • 状态模式的结构与实现都较为复杂,如果使用不当将导致程序结构和代码的混乱;
  • 状态模式对"开闭原则"的支持并不太好,对于可以切换状态的状态模式,增加新的状态类需要修改那些负责状态转换的源代码,否则无法切换到新增状态,而且修改某个状态类的行为也需修改对应类的源代码。

状态模式的适用场景:

  • 当行为需要随着状态的改变而改变时;
  • 当条件、分支语句很复杂冗长而需要优化时。

状态模式和策略模式:

  • 状态模式和策略模式的相同点是它们都需要根据需求选择相应的状态或策略;
  • 状态模式和策略模式的不同点是状态模式是在一个类中通过不同的动作切换不同的状态,而策略模式是为一个类选择某个策略,即状态模式中的Context是和多个状态关联的,而策略模式中的Context只和一个策略关联。

2、案例

  在这个例子中,我们要实现一个糖果机的功能:

  投入一元钱,转动摇把,就可以得到一颗糖果(当然是在及其中还有糖果的情况下);有10%的概率能够得到两颗糖果(被称为Winner)。具体代码如下:

  状态接口State中的代码:

public interface State {
// 投入一元钱
void onMoneyInserted(); // 将钱退回
void onMoneyReturned(); // 转动摇把
void onCrankTurned(); // 给出糖果
void onCandyPoped();
}

  没有投币的状态NoMoneyState中的代码:

public class NoMoneyState implements State {
private CandyMachine candyMachine; public NoMoneyState(CandyMachine candyMachine) {
this.candyMachine = candyMachine;
} @Override
public void onMoneyInserted() {
System.out.println("投币成功!");
candyMachine.setState(candyMachine.getMoneyInsertedState());
} @Override
public void onMoneyReturned() {
System.out.println("还没有投币!操作无效!");
} @Override
public void onCrankTurned() {
System.out.println("还没有投币!操作无效!");
} @Override
public void onCandyPoped() {
System.out.println("还没有投币!操作无效!");
}
}

  投币后的状态MoneyInsertedState中的代码:

public class MoneyInsertedState implements State {
private CandyMachine candyMachine; public MoneyInsertedState(CandyMachine candyMachine) {
this.candyMachine = candyMachine;
} @Override
public void onMoneyInserted() {
System.out.println("不能多次投币!");
} @Override
public void onMoneyReturned() {
System.out.println("正在退钱......退钱成功!");
candyMachine.setState(candyMachine.getNoMoneyState());
} @Override
public void onCrankTurned() {
System.out.println("转动了摇把......");
int random = (int) (Math.random() * 10000);
if (random % 10 == 0 && candyMachine.getCandyCount() > 1) {
System.out.println("恭喜您获得了两颗糖果!");
candyMachine.setState(candyMachine.getWinnerState());
candyMachine.popCandy();
} else {
if (candyMachine.getCandyCount() > 0) {
candyMachine.setState(candyMachine.getCandyPoppingState());
candyMachine.popCandy();
} else {
candyMachine.setState(candyMachine.getNoCandyState());
candyMachine.popCandy();
}
}
} @Override
public void onCandyPoped() {
System.out.println("请先转动摇把!");
}
}

  没有糖果的状态NoCandyState中的代码:

public class noCandyState implements State {
private CandyMachine candyMachine; public noCandyState(CandyMachine candyMachine) {
this.candyMachine = candyMachine;
} @Override
public void onMoneyInserted() {
System.out.println("已经没有糖果了!");
onMoneyReturned();
} @Override
public void onMoneyReturned() {
System.out.println("投币退回!");
candyMachine.setState(candyMachine.getNoMoneyState());
} @Override
public void onCrankTurned() {
System.out.println("已经没有糖果了!操作无效");
onMoneyReturned();
} @Override
public void onCandyPoped() {
System.out.println("已经没有糖果了!");
onMoneyReturned();
}
}

  正在发放糖果的状态CandyPoppingState中的代码:

public class CandyPoppingState implements State {
private CandyMachine candyMachine; public CandyPoppingState(CandyMachine candyMachine) {
this.candyMachine = candyMachine;
} @Override
public void onMoneyInserted() {
System.out.println("正在发放糖果,请勿操作!");
} @Override
public void onMoneyReturned() {
System.out.println("正在发放糖果,请勿操作!");
} @Override
public void onCrankTurned() {
System.out.println("正在发放糖果,请勿操作!");
} @Override
public void onCandyPoped() {
candyMachine.setCandyCount(candyMachine.getCandyCount() - 1);
System.out.println("成功发放了一颗糖果!");
candyMachine.setState(candyMachine.getNoMoneyState());
}
}

  获得两颗糖果的状态WinnerState中的代码:

public class WinnerState implements State {
private CandyMachine candyMachine; public WinnerState(CandyMachine candyMachine) {
this.candyMachine = candyMachine;
} @Override
public void onMoneyInserted() {
System.out.println("正在处理,请勿操作!");
} @Override
public void onMoneyReturned() {
System.out.println("正在处理,请勿操作!");
} @Override
public void onCrankTurned() {
System.out.println("正在处理,请勿操作!");
} @Override
public void onCandyPoped() {
candyMachine.setCandyCount(candyMachine.getCandyCount() - 2);
System.out.println("正在发放第一颗糖果......");
System.out.println("正在发放第二颗糖果......");
System.out.println("糖果发放成功!");
candyMachine.setState(candyMachine.getNoMoneyState());
}
}

  糖果机类CandyMachine中的代码:

public class CandyMachine {
private State moneyInsertedState;
private State noMoneyState;
private State noCandyState;
private State candyPoppingState;
private State winnerState; private State state;
private int candyCount; public CandyMachine(int candyCount) {
this.moneyInsertedState = new MoneyInsertedState(this);
this.noMoneyState = new NoMoneyState(this);
this.noCandyState = new noCandyState(this);
this.candyPoppingState = new CandyPoppingState(this);
this.winnerState = new WinnerState(this);
this.state = noMoneyState;
this.candyCount = candyCount;
} // 投币
public void insertMoney() {
this.state.onMoneyInserted();
} // 退钱
public void returnMoney() {
this.state.onMoneyReturned();
} // 转动摇把
public void turnCrank() {
this.state.onCrankTurned();
} // 发放糖果
public void popCandy() {
this.state.onCandyPoped();
} public State getMoneyInsertedState() {
return moneyInsertedState;
} public void setMoneyInsertedState(State moneyInsertedState) {
this.moneyInsertedState = moneyInsertedState;
} public State getNoMoneyState() {
return noMoneyState;
} public void setNoMoneyState(State noMoneyState) {
this.noMoneyState = noMoneyState;
} public State getNoCandyState() {
return noCandyState;
} public void setNoCandyState(State noCandyState) {
this.noCandyState = noCandyState;
} public State getCandyPoppingState() {
return candyPoppingState;
} public void setCandyPoppingState(State candyPoppingState) {
this.candyPoppingState = candyPoppingState;
} public State getWinnerState() {
return winnerState;
} public void setWinnerState(State winnerState) {
this.winnerState = winnerState;
} public State getState() {
return state;
} public void setState(State state) {
this.state = state;
} public int getCandyCount() {
return candyCount;
} public void setCandyCount(int candyCount) {
this.candyCount = candyCount;
}
}

  测试类Test中的代码:

public class Test {
public static void main(String[] args) {
CandyMachine candyMachine = new CandyMachine(5); candyMachine.insertMoney();
candyMachine.turnCrank(); candyMachine.insertMoney();
candyMachine.turnCrank(); candyMachine.insertMoney();
candyMachine.turnCrank(); candyMachine.insertMoney();
candyMachine.turnCrank(); candyMachine.insertMoney();
candyMachine.turnCrank(); candyMachine.insertMoney();
candyMachine.turnCrank();
}
}

  运行结果如下图所示(左图是每次都获得一颗糖果的情况;右图是有一次获得了两颗糖果的情况):

                                           

  最后贴出状态模式的GitHub代码地址:【GitHub - State】

【设计模式 - 20】之状态模式(State)的更多相关文章

  1. [设计模式-行为型]状态模式(State)

    一句话 在一个类的对象中维护状态的类的对象 概括

  2. 《JAVA设计模式》之状态模式(State)

    在阎宏博士的<JAVA与模式>一书中开头是这样描述状态(State)模式的: 状态模式,又称状态对象模式(Pattern of Objects for States),状态模式是对象的行为 ...

  3. 第20章 状态模式(State Pattern)

    原文 第20章 状态模式(State Pattern) 状态模式  概述:   当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类. 状态模式主要解决的是当控制一个对象状态的条件表 ...

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

    软件模式是将模式的一般概念应用于软件开发领域,即软件开发的 总体指导思路或参照样板.软件模式并非仅限于设计模式,还包括 架构模式.分析模式和过程模式等,实际上,在软件生存期的每一个阶段都存在着一些被认 ...

  5. 设计模式之 -- 状态模式(State)

     状态模式允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它的类.当控制一个对象的状态转换条件分支语句(if...else或switch...case)过于复杂时,可以此模式将状态的判断逻辑 ...

  6. 【转】设计模式 ( 十七) 状态模式State(对象行为型)

    设计模式 ( 十七) 状态模式State(对象行为型) 1.概述 在软件开发过程中,应用程序可能会根据不同的情况作出不同的处理.最直接的解决方案是将这些所有可能发生的情况全都考虑到.然后使用if... ...

  7. 设计模式 ( 十七) 状态模式State(对象行为型)

    设计模式 ( 十七) 状态模式State(对象行为型) 1.概述 在软件开发过程中,应用程序可能会根据不同的情况作出不同的处理.最直接的解决方案是将这些所有可能发生的情况全都考虑到.然后使用if... ...

  8. 乐在其中设计模式(C#) - 状态模式(State Pattern)

    原文:乐在其中设计模式(C#) - 状态模式(State Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 状态模式(State Pattern) 作者:webabcd 介绍 允 ...

  9. 北风设计模式课程---状态模式State(对象行为型)

    北风设计模式课程---状态模式State(对象行为型) 一.总结 一句话总结: 状态模式 具体状态的行为在具体的状态类中就解决,不用交给外部做判断.实质是将多条件判断弄成了多个类,在不同的类中做判断 ...

  10. 二十四种设计模式:状态模式(State Pattern)

    状态模式(State Pattern) 介绍允许一个对象在其内部状态改变时改变它的行为.对象看起来似乎修改了它所属的类. 示例有一个Message实体类,对它的操作有Insert()和Get()方法, ...

随机推荐

  1. [学习笔记]设计模式之Proxy

    为方便读者,本文已添加至索引: 设计模式 学习笔记索引 写在前面 “魔镜啊魔镜,谁是这个世界上最美丽的人?” 每到晚上,女王都会问魔镜相同的问题(见Decorator模式).这是她还曾身为女巫时留下的 ...

  2. 转载C#泛型集合—Dictionary<K,V>使用技巧

    1.要使用Dictionary集合,需要导入C#泛型命名空间 System.Collections.Generic(程序集:mscorlib) 2.描述 1).从一组键(Key)到一组值(Value) ...

  3. CentOS 6.4 64位 安装 apache-tomcat-6.0.43

    下载 tomcat: 地址:http://mirrors.hust.edu.cn/apache/tomcat/tomcat-6/v6.0.43/bin/apache-tomcat-6.0.43.tar ...

  4. C# 获取路径中文件名、目录、扩展名等

    string path = "C:\\dir1\\dir2\\foo.txt"; string str = "GetFullPath:" + Path.GetF ...

  5. WPF 进度条

    //Create a Delegate that matches the Signature of the ProgressBar's SetValue method private delegate ...

  6. mysql-积累管理sql语句

    //连接数据库 mysql -h xxx -u root -p; //查看数据库 show databases //查看数据表 show tables //查看某数据表结构 desc xxx表 //修 ...

  7. PHP获取当前页面完整url地址,包括参数的函数

    //php获取当前访问的完整url地址 function get_current_url(){     $current_url='http://';     if(isset($_SERVER['H ...

  8. JAVA面试题相关基础知识

        1.面向对象的特征有哪些方面 ①抽象: 抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面.抽象并不打算了解全部问题,而只是选择其中的一部分,暂时不用部分细节 ...

  9. Mac、Linux与Windows

    Mac本身是基于达尔文内核(Darwin内核),是苹果由UNIX改造的类UNIX,然后在这内核基础上搭建的图形界面 Linux确实是个好东西,你只需要一个键盘,一个显示器,一根网线,接入网络,便能做几 ...

  10. 《Journey》风之旅人;

    俩个人在茫茫世界相遇,互不相识,却能互相取暖,一路旅程,看尽了美丽的风景,也共同经历了暴风雪,然而该来的人会来,该走的人会走,这不就是人生旅途?