一 概念
自然界很多事物都有多种状态,而且不同状态下会具有不同的行为,这些状态在特定条件下还会发生相互转换,比如水

二 原理
在软件系统中,有些对象也像水一样具有多种状态,这些状态在某些情况下能够相互转换,而且对象在不同状态下也将具有不同的行为.
**状态模式(state pattern)的定义: 允许一个对象在其内部状态改变时改变它的行为. 对象看起来似乎修改了它的类.**
**状态模式**就是用于解决系统中复杂对象的状态转换以及不同状态下行为的封装问题. 状态模式将一个对象的状态从该对象中分离出来,封装到专门的状态类中(用类来表示状态) ,使得对象状态可以灵活变化.
三 结构图

四 应用场景
状态模式应用实例
模拟交通信号灯的状态转换. 交通信号灯一般包括了红、黄、绿3种颜色状态,不同状态之间的切换逻辑为: 红灯只能切换为黄灯,黄灯可以切换为绿灯或红灯,绿灯只能切换为黄灯.

五 实例
5.1 三种状态下的灯-具体状态下的类
/**
* 交通灯状态接口
**/
public interface State {
void switchToGreen(TrafficLight trafficLight); //切换为绿灯
void switchToYellow(TrafficLight trafficLight); //切换为黄灯
void switchToRed(TrafficLight trafficLight); //切换为红灯
}
/**
绿灯
**/
public class GreenState implements State {
@Override
public void switchToGreen(TrafficLight trafficLight) {
System.out.println("当前是绿灯,无需切换!");
}
@Override
public void switchToYellow(TrafficLight trafficLight) {
System.out.println("黄灯亮起...时长: 10秒");
}
@Override
public void switchToRed(TrafficLight trafficLight) {
System.out.println("绿灯不能够切换为红灯!");
}
}
/**
红灯
**/
public class RedState implements State {
@Override
public void switchToGreen(TrafficLight trafficLight) {
System.out.println("红灯不能切换为绿灯!");
}
@Override
public void switchToYellow(TrafficLight trafficLight) {
System.out.println("黄灯亮起...时长: 10秒");
}
@Override
public void switchToRed(TrafficLight trafficLight) {
System.out.println("当前为红灯,无需切换!");
}
}
/**
黄灯
**/
public class YellowState implements State {
@Override
public void switchToGreen(TrafficLight trafficLight) {
System.out.println("绿灯亮起...时长:60秒!");
}
@Override
public void switchToYellow(TrafficLight trafficLight) {
System.out.println("当前是黄灯,无需切换!");
}
@Override
public void switchToRed(TrafficLight trafficLight) {
System.out.println("红灯亮起...时长:90秒!");
}
}
5.2 控制状态的核心类 状态模式(state pattern)的定义: 允许一个对象在其内部状态改变时改变它的行为. 对象看起来似乎修改了它的类
/**
* 交通灯类
**/
public class TrafficLight {
//初始化-红灯
State state = new RedState();
public void setState(State state) {
this.state = state;
}
//切换为绿灯,通行状态
public void switchToGreen(){
state.switchToGreen(this);
}
//切换为黄灯,警示状态
public void switchToYellow() {
state.switchToYellow(this);
}
//切换为红灯,禁止状态
public void switchToRed(){
state.switchToRed(this);
}
}
5.3 调用
/**
**/
public class Client {
public static void main(String[] args) {
TrafficLight trafficLight = new TrafficLight();
trafficLight.state = new RedState();//切换状态
trafficLight.switchToRed();
trafficLight.switchToGreen();
trafficLight.switchToYellow();
}
}
六 总结
### 6.5.5 状态模式总结
**1) 状态模式的优点:**
* 将所有与某个状态有关的行为放到一个类中,并且可以方便地增加新的状态,只需要改变对象状态即可改变对象的行为。
* 允许状态转换逻辑与状态对象合成一体,而不是某一个巨大的条件语句块。
**2) 状态模式的缺点:**
* 状态模式的使用必然会增加系统类和对象的个数。
* 状态模式的结构与实现都较为复杂,如果使用不当将导致程序结构和代码的混乱。
* 状态模式对"开闭原则"的支持并不太好 (添加新的状态类需要修改那些负责状态转换的源代码)。
**3) 状态模式常见的使用场景:**
- 对象根据自身状态的变化来进行不同行为的操作时, 比如,购物订单状态。
- 对象需要根据自身变量的当前值改变行为,不期望使用大量 if-else 语句时, 比如,商品库存状态。
- 对于某些确定的状态和行为,不想使用重复代码时, 比如,某一个会员当天的购物浏览记录。