一、什么是状态模式
状态模式允许一个对象在其内部状态改变的时候改变其行为。这个对象看上去就像是改变了它的类一样。
二、状态模式应用场景
1.一个对象的行为取决于它的状态,并且它必须在运行时刻根据状态改变它的行为。
2.操作中含有庞大的多分支的条件语句,且这些分支依赖于该对象的状态。这个状态通常用一个或多个枚举常量表示。 通常,有多个操作包含这一相同的条件结构。State模式将每一个条件分支放入一个独立的类中。这使得你可以根据对象自身的情况将对象的状态作为一个对象,这一对象可以不依赖于其他对象而独立变化。
三、状态模式结构图
1、状态模式重点在各状态之间的切换从而做不同的事情,而策略模式更侧重于根据具体情况选择策略,并不涉及切换。
2、状态模式不同状态下做的事情不同,而策略模式做的都是同一件事,例如聚合支付平台,有支付宝、微信支付、银联支付,虽然策略不同,但最终做的事情都是支付,也就是说他们之间是可替换的。反观状态模式,各个状态的同一方法做的是不同的事,不能互相替换。
状态模式封装了对象的状态,而策略模式封装算法或策略。因为状态是跟对象密切相关的,它不能被重用;而通过从Context中分离出策略或算法,我们可以重用它们。
在状态模式中,每个状态通过持有Context的引用,来实现状态转移;但是每个策略都不持有Context的引用,它们只是被Context使用。
可以简单的将状态模式理解为同一个人做多件事,策略模式理解为不同的人做同一件事。
四、代码实现
需要重构的代码
public class OrderService {
public String orderState(String state) {
if (state.equals("0")) {
return "已经发货";
}
if (state.equals("1")) {
return "正在运送中...";
}
if (state.equals("2")) {
return "正在派送中...";
}
if (state.equals("3")) {
return "已经签收";
}
if (state.equals("4")) {
return "拒绝签收";
}
if (state.equals("5")) {
return "订单交易失败";
}
return "未找到对应的状态";
}
}
使用状态模式进行重构
OrderState定义统一抽象接口
public interface OrderState {
void orderService();
}
实现类
@Slf4j
@Component
public class ShippedAlreadyOrderState implements OrderState {
public void orderService() {
log.info(">>>切换为已经发货状态..");
}
}
@Slf4j
@Component
public class InTransitOrderState implements OrderState {
@Override
public void orderService() {
log.info(">>>切换为正在运送状态...");
}
}
@Slf4j
@Component
public class AlreadySignedOrderState implements OrderState {
@Override
public void orderService() {
log.info(">>>切换为已经签收状态..");
}
}
context上下文
public class ContextState {
private OrderState orderState;
public ContextState(OrderState orderState) {
this.orderState = orderState;
}
public void switchStateOrder() {
orderState.orderService();
}
}
controller
@RestController
public class StatusController {
@RequestMapping("/orderState")
public String orderState(String stateBeanId) {
OrderState orderState = SpringUtils.getBean(stateBeanId, OrderState.class);
ContextState contextState = new ContextState(orderState);
contextState.switchStateOrder();
return "success";
}
}
springUtils
@Component
public class SpringUtils implements ApplicationContextAware {
private static ApplicationContext applicationContext;
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.applicationContext = applicationContext;
}
//获取applicationContext
public static ApplicationContext getApplicationContext() {
return applicationContext;
}
//通过name获取 Bean.
public static Object getBean(String name){
return getApplicationContext().getBean(name);
}
//通过class获取Bean.
public static <T> T getBean(Class<T> clazz){
return getApplicationContext().getBean(clazz);
}
//通过name,以及Clazz返回指定的Bean
public static <T> T getBean(String name,Class<T> clazz){
return getApplicationContext().getBean(name, clazz);
}
}
启动类
@SpringBootApplication
public class StatusApp {
public static void main(String[] args) {
SpringApplication.run(StatusApp.class,args);
}
}
pom文件
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.0.1.RELEASE</version>
</parent>
<dependencies>
<!-- sprinboot web -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.16.10</version>
</dependency>
<dependency>
<groupId>commons-lang</groupId>
<artifactId>commons-lang</artifactId>
<version>2.6</version>
</dependency>
</dependencies>
测试结果
访问:
http://localhost:8080/orderState?stateBeanId=inTransitOrderState
2019-06-11 19:52:48.281 INFO 17192 — [nio-8080-exec-6] c.xwhy.service.impl.InTransitOrderState : >>>切换为正在运送状态…
访问:http://localhost:8080/orderState?stateBeanId=shippedAlreadyOrderState
2019-06-11 19:52:02.077 INFO 17192 — [nio-8080-exec-4] c.x.s.impl.ShippedAlreadyOrderState : >>>切换为已经发货状态…
访问:http://localhost:8080/orderState?stateBeanId=alreadySignedOrderState
2019-06-11 19:50:28.310 INFO 17192 — [nio-8080-exec-1] c.x.s.impl.AlreadySignedOrderState : >>>切换为已经签收状态…