状态机(State Machine)是一种设计模式,用于描述对象在不同状态之间的转换和行为。状态机可以帮助开发者管理复杂的状态逻辑,使得系统在不同状态下的行为更易于理解和维护。以下是关于状态机设计模式的详细介绍。
1. 状态机的基本概念
- 状态:表示对象在某一时刻的情况或条件。例如,订单的状态可以是"新建"、"处理中"、"已完成"。
- 事件:导致状态变化的触发器,例如用户操作、时间到达或其他外部输入。
- 状态转移:从一个状态到另一个状态的过程,通常由事件驱动。
- 上下文:持有状态机当前状态及相关数据的对象。
2. 状态机的组成部分
状态机通常由以下几个组成部分构成:
- 状态(States):定义了系统可能处于的所有状态。
- 事件(Events):触发状态变迁的事件。
- 转移(Transitions):描述状态之间的转换规则,通常与特定事件关联。
- 行为(Actions):在状态进入、退出或转移时执行的操作。
3. 状态机的类型
状态机可以分为两种主要类型:
- 有限状态机(Finite State Machine, FSM):状态数量有限,适用于大多数场景。
- 层次状态机(Hierarchical State Machine):允许状态嵌套,可以更好地组织复杂的状态逻辑。
4. 状态机的优点
- 清晰性:将状态和行为明确分开,使代码更加易读和可维护。
- 灵活性:便于修改和扩展状态逻辑,例如添加新的状态或事件。
- 可测试性:每个状态和转移都可以独立测试,提高了系统的可靠性。
5. 状态机的实现
状态机可以通过多种方式实现,常见的方法包括:
-
使用条件语句(if-else 或 switch-case):简单的状态机可以用条件语句直接实现,但随着状态和事件的增加,代码会变得复杂且难以维护。
-
状态模式(State Pattern):一种面向对象的设计模式,通过创建状态类来封装状态相关的行为,实现动态状态切换。
-
状态机框架 :使用现有的状态机库或框架(如
Stateless4j
、Spring State Machine
等)来简化状态机的创建和管理。
6. 状态模式示例
下面是一个使用状态模式实现简单状态机的示例,展示如何管理订单的状态。
java
// 定义状态接口
interface OrderState {
void handle(OrderContext context);
}
// 具体状态实现
class NewOrderState implements OrderState {
public void handle(OrderContext context) {
System.out.println("Handling new order.");
context.setState(new ProcessingOrderState());
}
}
class ProcessingOrderState implements OrderState {
public void handle(OrderContext context) {
System.out.println("Processing order.");
context.setState(new CompletedOrderState());
}
}
class CompletedOrderState implements OrderState {
public void handle(OrderContext context) {
System.out.println("Order completed.");
}
}
// 上下文类
class OrderContext {
private OrderState state;
public OrderContext() {
this.state = new NewOrderState(); // 初始状态
}
public void setState(OrderState state) {
this.state = state;
}
public void request() {
state.handle(this);
}
}
// 客户端代码
public class StatePatternDemo {
public static void main(String[] args) {
OrderContext order = new OrderContext();
order.request(); // 处理新订单
order.request(); // 处理中的订单
order.request(); // 订单已完成
}
}
7. 总结
状态机设计模式是管理复杂状态和行为的一种有效方法。通过将状态、事件和转移逻辑清晰地分开,状态机使得程序的结构更加清晰,易于理解和维护。无论是在游戏开发、工作流管理还是其他需要状态管理的应用中,状态机都是一种非常有用的工具。使用现有的状态机框架可以进一步简化开发过程,提高效率。
轻量级状态机stateless4j的使用
-
可以把当前状态和状态字段持久化的数据库里,实例中存储到文件里
resources/stateless.json[
{
"state": "待付款",
"trigger": "用户付款",
"destinationState": "待发货"
},
{
"state": "待发货",
"trigger": "发货",
"destinationState": "待收货"
},
{
"state": "待收货",
"trigger": "确认收货",
"destinationState": "已完成"
},
{
"state": "已完成",
"trigger": "评价",
"destinationState": "已评价"
},
{
"state": "已评价",
"trigger": "删除",
"destinationState": "已删除"
}
]
测试用例
/**
* 状态从file中读取
*/
@Test
public void stateFromDb() throws IOException {
String content = FileUtils.readToEnd(
StateMachineExample.class.getClassLoader().getResourceAsStream("stateless.json"),
Charset.defaultCharset());
StateMachineConfig<String, String> stateMachineConfig = new StateMachineConfig<>();
List<Map<String, String>> map = objectMapper.readValue(content, new TypeReference<List<Map<String, String>>>() {
});
map.forEach(o -> {
String state = o.get("state");
String trigger = o.get("trigger");
String destinationState = o.get("destinationState");
stateMachineConfig.configure(state).permit(trigger, destinationState);
});
StateMachine<String, String> stateMachine = new StateMachine<>("待收货", stateMachineConfig);
log.info("当前状态:{}", stateMachine.getState());
stateMachine.fire("确认收货");
log.info("当前状态:{}", stateMachine.getState());
}
结果
如果你的state和trigger不一致,比如状态是待收获
,你的动作是评价
,那状态机就会报错,状态不能正确更新