Springboot 使用设计模式- 策略模式
前言
直白点,什么场景我们需要使用到设计模式- 策略模式。
在平常的springboot项目里面做CRUD,我们的习惯性基本是 一个mapper,一个service,一个serviceImpl。
但是有时候会出现一种情况,就是实现类impl里面虽然业务主线是归为一致的,但是需要进行类别判断去执行不同的业务逻辑,
导致impl里面出现 一个类别的if 后面跟上一大段业务代码,然后同时存在这一几个 if 判断,这种情况恨不得单独把代码抽出去写成方法,看起来稍微舒服一些(其实这种情况使用策略模式是非常适合的)。
举例场景:
我现在需要实现一个关于支付预下单的功能,然后这个预下单功能大致用到的参数都一样,
但是在下单类型上分为 国内单,海外单,折扣单。
三种不同的下单流程都属于预下单功能的一部分,但是结合业务后,逻辑算法都是基本独立的。
按照往常的写法,可能会出现如下的代码写法:
或者有的为了不想麻烦,直接就建了三个service,然后三个serviceImpl。
那么就针对上面这个场景,我们使用设计模式-策略模式,就能非常润滑地去实现。
正文
订单信息实体类,OrderInfo.java:
import lombok.Data;
/**
* @Author : JCccc
* @CreateTime : 2020/5/11
* @Description :
**/
@Data
public class OrderInfo {
private String orderId; //订单id
private String platFormType; //(平台)订单类型
private Double amount; //金额
private String createTime; //创建时间
//....省略若干业务字段
}
预下单的策略接口,OrderStrategyService.java :
/**
* @Author : JCccc
* @CreateTime : 2020/5/11
* @Description :
**/
public interface OrderStrategyService {
//预下单
String preCreateOrder(OrderInfo orderInfo);
}
接下来是三个类型订单的实现类,都实现策略接口:
国内,OrderDomestic.java:
/**
* @Author : JCccc
* @CreateTime : 2020/5/11
* @Description :国内
**/
@Component("Domestic")
public class OrderDomestic implements OrderStrategyService {
@Override
public String preCreateOrder(OrderInfo orderInfo) {
System.out.println("*处理国内预下单的相关业务*");
return orderInfo.getPlatFormType()+"-国内预下单";
}
}
海外,OrderOverseas.java :
/**
* @Author : JCccc
* @CreateTime : 2020/5/11
* @Description :海外
**/
@Component("Overseas")
public class OrderOverseas implements OrderStrategyService {
@Override
public String preCreateOrder(OrderInfo orderInfo) {
System.out.println("**处理海外预下单的相关业务**");
return orderInfo.getPlatFormType()+"-海外预下单";
}
}
特殊回扣,
/**
* @Author : JCccc
* @CreateTime : 2020/5/11
* @Description :特殊订单(回扣)
**/
@Component("Rebate")
public class OrderRebate implements OrderStrategyService {
@Override
public String preCreateOrder(OrderInfo orderInfo) {
System.out.println("***处理国内特殊回扣预下单的相关业务***");
return orderInfo.getPlatFormType()+"-特殊回扣预下单";
}
}
可以看到上面三个实现类都实现了策略接口OrderStrategyService,而且使用了注解@Component +组件名。
那么接下来就是策略设计模式的核心, 策略接口实现类的调度使用类,OrderStrategyContext.java(留意看里面的注释):
/**
* @Author : JCccc
* @CreateTime : 2020/5/11
* @Description : 利用Spring的发现机制,将实现了OrderStrategyService的类都put到orderStrategyMap里面。
* 后面只需要根据platformId对应好 各个实现类的注解 如: @Component("Domestic") 就可以取出不同的业务实现类
**/
@Service
public class OrderStrategyContext {
private final Map<String, OrderStrategyService> orderStrategyMap = new ConcurrentHashMap<>();
public OrderStrategyContext(Map<String, OrderStrategyService> strategyMap) {
this.orderStrategyMap.clear();
strategyMap.forEach((k, v)-> this.orderStrategyMap.put(k, v));
}
public OrderStrategyService getResource(OrderInfo orderInfo){
return orderStrategyMap.get(orderInfo.getPlatFormType());
}
}
ps: 可以看到getResource这个方法,获取资源,里面其实就是根据订单类型去取出对应的业务实现类。
ok,到这策略模式的结合使用已经大致完毕,最后是通过策略调度类去使用我们不同类型的实现类。
写一个简单的接口,看看效果:
OrderTestController.java :
/**
* @Author : JCccc
* @CreateTime : 2020/5/11
* @Description :
**/
@RestController
public class OrderTestController {
@Autowired
private OrderStrategyContext orderStrategyContext;
@PostMapping("/testStrategy")
public String testStrategy(@RequestBody OrderInfo orderInfo){
OrderStrategyService orderServiceImpl = orderStrategyContext.getResource(orderInfo);
String resultTest = orderServiceImpl.preCreateOrder(orderInfo);
return resultTest;
}
}
特殊回扣下单:
海外下单:
可以看到控制台: