【设计模式 - 20】之状态模式(State)
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)的更多相关文章
- [设计模式-行为型]状态模式(State)
一句话 在一个类的对象中维护状态的类的对象 概括
- 《JAVA设计模式》之状态模式(State)
在阎宏博士的<JAVA与模式>一书中开头是这样描述状态(State)模式的: 状态模式,又称状态对象模式(Pattern of Objects for States),状态模式是对象的行为 ...
- 第20章 状态模式(State Pattern)
原文 第20章 状态模式(State Pattern) 状态模式 概述: 当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类. 状态模式主要解决的是当控制一个对象状态的条件表 ...
- 状态模式(State)-设计模式
软件模式是将模式的一般概念应用于软件开发领域,即软件开发的 总体指导思路或参照样板.软件模式并非仅限于设计模式,还包括 架构模式.分析模式和过程模式等,实际上,在软件生存期的每一个阶段都存在着一些被认 ...
- 设计模式之 -- 状态模式(State)
状态模式允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它的类.当控制一个对象的状态转换条件分支语句(if...else或switch...case)过于复杂时,可以此模式将状态的判断逻辑 ...
- 【转】设计模式 ( 十七) 状态模式State(对象行为型)
设计模式 ( 十七) 状态模式State(对象行为型) 1.概述 在软件开发过程中,应用程序可能会根据不同的情况作出不同的处理.最直接的解决方案是将这些所有可能发生的情况全都考虑到.然后使用if... ...
- 设计模式 ( 十七) 状态模式State(对象行为型)
设计模式 ( 十七) 状态模式State(对象行为型) 1.概述 在软件开发过程中,应用程序可能会根据不同的情况作出不同的处理.最直接的解决方案是将这些所有可能发生的情况全都考虑到.然后使用if... ...
- 乐在其中设计模式(C#) - 状态模式(State Pattern)
原文:乐在其中设计模式(C#) - 状态模式(State Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 状态模式(State Pattern) 作者:webabcd 介绍 允 ...
- 北风设计模式课程---状态模式State(对象行为型)
北风设计模式课程---状态模式State(对象行为型) 一.总结 一句话总结: 状态模式 具体状态的行为在具体的状态类中就解决,不用交给外部做判断.实质是将多条件判断弄成了多个类,在不同的类中做判断 ...
- 二十四种设计模式:状态模式(State Pattern)
状态模式(State Pattern) 介绍允许一个对象在其内部状态改变时改变它的行为.对象看起来似乎修改了它所属的类. 示例有一个Message实体类,对它的操作有Insert()和Get()方法, ...
随机推荐
- 网站访问架构cdn与负载均衡
曾经见到知乎上有人问“为什么像facebook这类的网站需要上千个工程师维护?”,下面的回答多种多样,但总结起来就是:一个高性能的web系统需 要从无数个角度去考虑他,大到服务器的布局,小到软件中某个 ...
- windows下node.js+sublime中安装coffeescript
node.js中安装Coffeescript 1.我的node.js安装目录 2.node.js 全局模块所在目录 3.node.js安装coffeescript npm install -g c ...
- sublime text3-代码片段配置
1.Tools->New Snippet-> <snippet> <content><![CDATA[${1:public }function ${2 ...
- apache2.4配置虚拟主机
step1 启用 httpd-vhosts.conf 找到E:/apache/Apache24/conf 中httpd.conf 文件,取消注释下面这句话 step2 在 httpd-vhosts.c ...
- ExecuteReader
最近在做winform的编程,想到一真没有使用过ExecuteReader.可能以前以后它的用户不大,或者 不大好用,故没有用过.今天在这里将学习记录写下来,供读者参考: 1.MSDN上说:Sends ...
- max os 安装python模块PIL
下载libjpeg和zlib: http://www.ijg.org/files/jpegsrc.v9.tar.gz http://zlib.net/zlib-1.2.8.tar.gz 安装libjp ...
- Tomcat启动分析(Tomcat7.0)
1)bin目录下的bootstrap.jar中的main方法启动Tomcat org.apache.catalina.startup.Bootstrap类下的main方法 可以看到Bootstrap类 ...
- 搭建Tornado+Nginx
Tornado一个高效的异步非阻塞式的实时Web服务器,是Facebook旗下的 FriendFeed 网站开源Web服务器版本.但是它内置的HTTP服务器功能有限,不能在生产环境下使用. 在 Fri ...
- C#隐私信息(银行账户,身份证号码,名字)中间部分特殊字符替换(*)
最近做到一个关于银行的一个功能模块,需要将隐私信息银行账号中间部分用*代替,于是写下了,如下代码: /// <summary> /// 将传入的字符串中间部分字符替换成特殊字符 /// & ...
- [BZOJ 1004] [HNOI2008] Cards 【Burnside引理 + DP】
题目链接:BZOJ - 1004 题目分析 首先,几个定义和定理引理: 群:G是一个集合,*是定义在这个集合上的一个运算. 如果满足以下性质,那么(G, *)是一个群. 1)封闭性,对于任意 a, b ...