需求
在流程管理中,通常是设计好一个流程模板,然后通过这个模板生成审批实例。
在设计模板时审批人一般是固定的,但是也可设计成一种审批类型,比如审批人设置为申请人的leader,或者申请所属项目的维护人员,这时候我们需要根据不同的审批类型设置不同的审批人。
很显然,我们的代码可以写成下面这个样子
代码语言:javascript复制List<String> approvers = new ArrayList<>();
if (approveType == Type.LEADER){
// 获取审批人逻辑
} else if () {
// ...
}
如果我们想增加审批类型,可以在后面继续写判断条件,在条件里边执行获取审批人逻辑,时间一长,代码越来越长,这样缺点存在两点
- 不易维护,代码很长,自己想要修改的地方只存在一个判断条件里,而你可能在获取审批人前边也写了很多代码,这时需要逐行阅读代码定位到需要修改的地方
- 不易扩展,如果想要增加一种审批类型,那么只能继续增加判断条件,导致不易维护的问题更加严重
- 效率低,判断条件过多影响程序执行效率
可以使用策略模式对代码进行重构
可以想到,灵活多变的是获取审批人的方式,我们参考设计模式的开闭原则,对获取方式进行开发,对获取流程封闭。
我们设计一个SpecialApprover
接口,增加一个获取审批人的方法,每个审批类型对应一个该接口的实现类。使用时,将实现类放入到map中,根据不同的审批类型获取不同的实现类,执行不同的获取审批人的逻辑。可以看到,我们在获取特定审批人时只需要一行代码。
Class {
@Autowired
@Qualifier("leaderApprove")
SpecialApprove leaderApprove;
@Autowired
@Qualifier("projectOpsApprove")
SpecialApprove projectOpsApprove;
Map<String, SpecialApprove> map;
init() {
map = new HashMap<>();
map.put(SpecialApproverEnum.LEADER.getValue(), leaderApprove);
map.put(SpecialApproverEnum.PROJECT_OPS.getValue(), projectOpsApprove);
}
void someAction(Map<String, Object> condition){
// ... 根据审批类型获取处理类,拿到审批人
List<String> approvers = map.get(Type).getApprovers(condition);
// ...
}
}
Interface SpecialApprover {
List<String> getApprovers(Map<String, Object> condition);
}
// 实现接口,覆写方法
Class LeaderApprover implments SpecialApprover {
@Override
List<String> getApprovers(Map<String, Object> condition){
/...
}
}
Class ProjectOpsApprover implments SpecialApprover {
@Override
List<String> getApprovers(Map<String, Object> condition){
/...
}
}
通过运用策略模式,有以下优点
- 易维护,如果需要修改某个审批类型获取审批人的逻辑,直接找到相应的类修改
- 方便扩展,很方便加入一个新的审批类型, 只需要实现接口,并将实现类加入到map中
- 执行效率高