声明:转载请说明来源:http://www.cnblogs.com/pony1223/p/7518226.html
一、引出状态模式
假设我们现在有一个糖果机项目,那么我们知道正常一般糖果机提供给用户的行为有这么几种:投入硬币、转动曲柄、退出硬币几种行为;那么糖果机呢一般有这几中状态,待机状态、持有硬币的准备状态、运行状态即正在售出状态和初始状态 这么几种正常状态。 我们发现处于不同状态的时候,持有的行为是不一样的,图如下:
如果我们采用传统的方法来写代码,那么在投入硬币这个行为操作的时候,我们会进行状态的判断,只有在处于待机状态情况下这种行为是正常的,而其他则非正常,那么其他行为也一样,都需要去先判断下当前的状态来进行操作。得到的代码则为:
1 packagestudy.designmode.statemode;2
3 public classCandyMachine {4
5 final static int SoldOutState = 0; //初始状态
6 final static int OnReadyState = 1; //待机状态
7 final static int HasCoin = 2; //准备状态
8 final static int SoldState = 3; //售出状态
9
10 private int state = SoldOutState; //变量,用于存放当前的状态值
11 private int count = 0; //糖果的数目
12
13 public CandyMachine(intcount) {14 this.count =count;15 if (count > 0) {16 state =OnReadyState;17 }18 }19
20 //投入硬币行为的时候,通过判断当前的状态来匹配所有的状态.
21 public voidinsertCoin() {22 switch(state) {23 caseSoldOutState:24 System.out.println("you can't insert coin,the machine sold out!");25 break;26 case OnReadyState: //只有在待机状态的时候,投入硬币行为正确,并将状态改变为准备状态
27 state =HasCoin;28 System.out29 .println("you have inserted a coin,next,please turn crank!");30 break;31 caseHasCoin:32 System.out.println("you can't insert another coin!");33
34 break;35 caseSoldState:36 System.out.println("please wait!we are giving you a candy!");37
38 break;39 }40
41 }42
43 //回退硬币
44 public voidreturnCoin() {45 switch(state) {46 caseSoldOutState:47 System.out48 .println("you can't return,you haven't inserted a coin yet!");49 break;50 caseOnReadyState:51 System.out.println("you haven't inserted a coin yet!");52 break;53 caseHasCoin:54
55 System.out.println("coin return!");56 state =OnReadyState;57
58 break;59 caseSoldState:60 System.out.println("sorry,you already have turned the crank!");61
62 break;63 }64
65 }66
67 //转动曲柄
68 public voidturnCrank() {69 switch(state) {70 caseSoldOutState:71 System.out.println("you turned,but there are no candies!");72 break;73 caseOnReadyState:74 System.out.println("you turned,but you haven't inserted a coin!");75 break;76 caseHasCoin:77 System.out.println("crank turn...!");78 state =SoldState;79 dispense();80 break;81 caseSoldState:82 System.out83 .println("we are giving you a candy,turning another get nothing,!");84 break;85 }86
87 }88
89 //触发发放糖果行为
90 private voiddispense() {91 count = count - 1;92 System.out.println("a candy rolling out!");93 if (count > 0) {94 state =OnReadyState;95 } else{96 System.out.println("Oo,out of candies");97 state =SoldOutState;98 }99
100 }101
102 public voidprintstate() {103
104 switch(state) {105 caseSoldOutState:106 System.out.println("***SoldOutState***");107 break;108 caseOnReadyState:109 System.out.println("***OnReadyState***");110 break;111 caseHasCoin:112
113 System.out.println("***HasCoin***");114
115 break;116 caseSoldState:117 System.out.println("***SoldState***");118 break;119 }120
121 }122 }
那么上面这种方式存在什么问题呢?首先很直观的感受就是:
1.存在大量的switch case 语句 当然可以用if else 也是一样的。
2.可扩展性差,并且一旦要加入一种新的状态,那么就会要修改所有的switch case 不符合开闭原则
3.没有采用面向对象的方式去封装
比如,这个时候,新增加了一种状态,赢家状态,即可以获取到两粒糖果;那么如果用上面的方式,肯定是不符合开闭原则的,同时扩展性也是不好的;那么我们有什么其它的方式来解决呢?
二、解决办法
为了解决上面的问题,我们首先分析项目中变化的部分和不变的部分,抽化出变化的部分,我们发现糖果机提供的行为一般是不变的,就是投入硬币、转动曲柄给、退回硬币、机器发放糖果;而糖果机的状态是可以变化的,可以新增出一种状态来,比如我们说的赢家状态。那么我们这个抽出变化的部分,即我们说的状态,于是出现了下面的结构设计方案:
这个结构图告诉我们,提炼出状态接口出来,然后将各个状态抽出,并去实现接口,每个状态都持有投入硬币,退回硬币,转动曲柄、售出糖果这几种行为对应的方法做出相应;而糖果机持有所有的状态,并通过引用状态接口来操作各个状态;这种设计架构就是我们说的状态模式。
状态模式定义:对象行为的变化是由于状态的变化引入,那么即当内部状态发生变化的时候,就会改变对象的行为,而这种改变视乎就改变了整个类。
那么现在采用状态模式来解决问题:
1.首先定义接口:
packagestudy.designmode.statemode.state;public interfaceState {public voidinsertCoin();public voidreturnCoin();public voidturnCrank();public voiddispense();public voidprintstate();
}
2.定义各个状态的实现类
准备状态:
packagestudy.designmode.statemode.state;importjava.util.Random;public class HasCoin implementsState {privateCandyMachine mCandyMachine;publicHasCoin(CandyMachine mCandyMachine) {this.mCandyMachine =mCandyMachine;
}
@Overridepublic voidinsertCoin() {//TODO Auto-generated method stub
System.out.println("you can't insert another coin!");
}
@Overridepublic voidreturnCoin() {//TODO Auto-generated method stub
System.out.println("coin return!");
mCandyMachine.setState(mCandyMachine.mOnReadyState);
}
@Overridepublic voidturnCrank() {//TODO Auto-generated method stub
System.out.println("crank turn...!");
Random ranwinner=newRandom();int winner=ranwinner.nextInt(10);if(winner==0)
{
mCandyMachine.setState(mCandyMachine.mWinnerState);
}else{
mCandyMachine.setState(mCandyMachine.mSoldState);
}
}
@Overridepublic voiddispense() {
}
@Overridepublic voidprintstate() {//TODO Auto-generated method stub
System.out.println("***HasCoin***");
}
}
说明:我们会发现里面存在一个糖果机的属性,而之所以存在这个属性,就是因为糖果机中持有所有的状态,而在准备状态下,肯定会由于某种行为发生状态改变,而要改变的状态都在糖果机中,所以持有一个糖果机属性,下面也一样,不在重复说明。
准备状态:
packagestudy.designmode.statemode.state;public class OnReadyState implementsState {privateCandyMachine mCandyMachine;publicOnReadyState(CandyMachine mCandyMachine)
{this.mCandyMachine=mCandyMachine;
}
@Overridepublic voidinsertCoin() {//TODO Auto-generated method stub
System.out
.println("you have inserted a coin,next,please turn crank!");
mCandyMachine.setState(mCandyMachine.mHasCoin);
}
@Overridepublic voidreturnCoin() {//TODO Auto-generated method stub
System.out.println("you haven't inserted a coin yet!");
}
@Overridepublic voidturnCrank() {//TODO Auto-generated method stub
System.out.println("you turned,but you haven't inserted a coin!");
}
@Overridepublic voiddispense() {//TODO Auto-generated method stub
}
@Overridepublic voidprintstate() {//TODO Auto-generated method stub
System.out.println("***OnReadyState***");
}
}
初始状态:
packagestudy.designmode.statemode.state;public class SoldOutState implementsState {privateCandyMachine mCandyMachine;publicSoldOutState(CandyMachine mCandyMachine)
{this.mCandyMachine=mCandyMachine;
}
@Overridepublic voidinsertCoin() {//TODO Auto-generated method stub
System.out.println("you can't insert coin,the machine sold out!");
}
@Overridepublic voidreturnCoin() {//TODO Auto-generated method stub
System.out
.println("you can't return,you haven't inserted a coin yet!");
}
@Overridepublic voidturnCrank() {//TODO Auto-generated method stub
System.out.println("you turned,but there are no candies!");
}
@Overridepublic voiddispense() {//TODO Auto-generated method stub
}
@Overridepublic voidprintstate() {//TODO Auto-generated method stub
System.out.println("***SoldOutState***");
}
}
售出状态:
packagestudy.designmode.statemode.state;public class SoldState implementsState {privateCandyMachine mCandyMachine;publicSoldState(CandyMachine mCandyMachine)
{this.mCandyMachine=mCandyMachine;
}
@Overridepublic voidinsertCoin() {//TODO Auto-generated method stub
System.out.println("please wait!we are giving you a candy!");
}
@Overridepublic voidreturnCoin() {//TODO Auto-generated method stub
System.out.println("you haven't inserted a coin yet!");
}
@Overridepublic voidturnCrank() {//TODO Auto-generated method stub
System.out
.println("we are giving you a candy,turning another get nothing,!");
}
@Overridepublic voiddispense() {//TODO Auto-generated method stub
mCandyMachine.releaseCandy();if (mCandyMachine.getCount() > 0) {
mCandyMachine.setState(mCandyMachine.mOnReadyState);
}else{
System.out.println("Oo,out of candies");
mCandyMachine.setState(mCandyMachine.mSoldOutState);
}
}
@Overridepublic voidprintstate() {//TODO Auto-generated method stub
System.out.println("***SoldState***");
}
}
赢家状态:
packagestudy.designmode.statemode.state;public class WinnerState implementsState {privateCandyMachine mCandyMachine;publicWinnerState(CandyMachine mCandyMachine) {this.mCandyMachine =mCandyMachine;
}
@Overridepublic voidinsertCoin() {//TODO Auto-generated method stub
System.out.println("please wait!we are giving you a candy!");
}
@Overridepublic voidreturnCoin() {//TODO Auto-generated method stub
System.out.println("you haven't inserted a coin yet!");
}
@Overridepublic voidturnCrank() {//TODO Auto-generated method stub
System.out
.println("we are giving you a candy,turning another get nothing,!");
}
@Overridepublic voiddispense() {//TODO Auto-generated method stub
mCandyMachine.releaseCandy();if (mCandyMachine.getCount() == 0) {
mCandyMachine.setState(mCandyMachine.mSoldOutState);
}else{
System.out.println("you are a winner!you get another candy!");
mCandyMachine.releaseCandy();if (mCandyMachine.getCount() > 0) {
mCandyMachine.setState(mCandyMachine.mOnReadyState);
}else{
System.out.println("Oo,out of candies");
mCandyMachine.setState(mCandyMachine.mSoldOutState);
}
}
}
@Overridepublic voidprintstate() {//TODO Auto-generated method stub
System.out.println("***WinnerState***");
}
}
3.糖果机,糖果机要持有所有的状态,并在初始化的时候,要设置其开始的状态,然后糖果的各个行为,就委托到了各个状态中自己维护,代码如下:
packagestudy.designmode.statemode.state;public classCandyMachine {
State mSoldOutState;
State mOnReadyState;
State mHasCoin;
State mSoldState;
State mWinnerState;privateState state;private int count = 0;public CandyMachine(intcount) {this.count =count;
mSoldOutState= new SoldOutState(this);
mOnReadyState= new OnReadyState(this);
mHasCoin= new HasCoin(this);
mSoldState= new SoldState(this);
mWinnerState= new WinnerState(this);if (count > 0) {
state=mOnReadyState;
}else{
state=mSoldOutState;
}
}public voidsetState(State state) {this.state =state;
}public voidinsertCoin() {
state.insertCoin();
}public voidreturnCoin() {
state.returnCoin();
}public voidturnCrank() {
state.turnCrank();
state.dispense();
}voidreleaseCandy() {//TODO Auto-generated method stub
if (count > 0) {
count= count - 1;
System.out.println("a candy rolling out!");
}
}public intgetCount() {returncount;
}public voidprintstate() {
state.printstate();
}
}
4.测试类
packagestudy.designmode.statemode.state;public classMainTest {public static voidmain(String[] args) {
CandyMachine mCandyMachine= new CandyMachine(6);
mCandyMachine.printstate();
mCandyMachine.insertCoin();
mCandyMachine.printstate();
mCandyMachine.turnCrank();
mCandyMachine.printstate();
mCandyMachine.insertCoin();
mCandyMachine.printstate();
mCandyMachine.turnCrank();
mCandyMachine.printstate();
}
}
结果如下:
可以和开始的传统方案对比,结果是一样的,但是具备了可扩展性。
三、总结
通过上面的例子,我们已经对状态模式有所了解,下面我们做一个总结,来回顾我们的状态模式:
1.状态模式允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它的类。
理解:这个模式将状态封装成独立的类,并将动作委托到代表当前状态的对象,这就是说行为会随着内部状态而改变。
“看起来好像修改了它的类”是什么意思呢?从客户的视角来看:如果说你使用的对象能够完全改变它的行为,那么你会觉得,这个对象实际上是从别的类实例化而来的。然而,实际上,你知道我们是在使用组合通过简单引用不同的状态对象来造成类改变的假象
2.状态模式要点
(1)客户不会和状态进行交互,全盘了解状态是 context的工作
(2)在状态模式中,每个状态通过持有Context的引用,来实现状态转移
(3)使用状态模式总是会增加设计中类的数目,这是为了要获得程序可扩展性,弹性的代价,如果你的代码不是一次性的,后期可能会不断加入不同的状态,那么状态模式的设计是绝对值得的。【同时也是一个缺点】
(4)状态类可以被多个context实例共享
3.状态模式和策略模式对比
首先让我们来看看它们之间更多的相似之处:
添加新的状态或策略都很容易,而且不需要修改使用它们的Context对象。
它们都让你的代码符合OCP原则(软件对扩展应该是开发的,对修改应该是关闭的)。在状态模式和策略模式中,Context对象对修改是关闭的,添加新的状态或策略,都不需要修改Context。
正如状态模式中的Context会有初始状态一样,策略模式同样有默认策略。
状态模式以不同的状态封装不同的行为,而策略模式以不同的策略封装不同的行为。
它们都依赖子类去实现相关行为
两个模式的差别在于它们的”意图“不同:
状态模式帮助对象管理状态,我们将一群行为封装早状态对象中,context的行为随时可委托到那些状态中的一个.随着时间的流逝,当前状态在状态对象集合中游走改变,以反映context内部状态,因此,context的行为也会跟着改变。当要添加新的状态时,不需要修改原来代码添加新的状态类即可。 而策略模式允许Client选择不同的行为。通过封装一组相关算法,为Client提供运行时的灵活性。Client可以在运行时,选择任一算法,而不改变使用算法的Context。一些流行的策略模式的例子是写那些使用算法的代码,例如加密算法、压缩算法、排序算法。客户通常主动指定context所要组合的策略对象是哪一个.
一句话:最根本的差异在于策略模式是在求解同一个问题的多种解法,这些不同解法之间毫无关联;状态模式则不同,状态模式要求各个状态之间有所关联,以便实现状态转移。