在阎宏博士的《JAVA与模式》一书中开头是这样描述策略(Strategy)模式的:
策略模式属于对象的行为模式。其用意是针对一组算法,将每一个算法封装到具有共同接口的独立的类中,从而使得它们可以相互替换。策略模式使得算法可以在不影响到客户端的情况下发生变化。
1.策略模式核心
- 策略调度与执行者:持有一个对象的引用,或者有办法拿到对象。执行对应策略。
- 抽象策略对象:这是一个抽象角色,通常由一个接口或抽象类实现。此角色给出所有的具体策略类所需的接口。
- 具体策略角色:继承或实现抽象策略对象,实现定义的方法,方法包装了相关的算法或行为。
2.原始代码存在的问题
难看,可扩展性差,不好维护,可读性比较差。这种代码写出去,后面接手的肯定得吐槽了。
@Autowired
private CompositeCpaCallBack compositeCpaCallBack;
----代码片段---
//改造前调用
//Boolean callResult = callbackAdvertisers(latestActiveClick);
//改造后调用
Boolean callResult = compositeCpaCallBack.callbackAdvertisers(latestActiveClick);
private Boolean callbackAdvertisers(CpaUnionInfo cpaUnionInfo) {
switch (CpaSourceEnum.valueOf(cpaUnionInfo.getSource())) {
case wangxiang:
。。。
break;
case jinritoutiao:
。。。
break;
case guangdiantong:
。。。
break;
case tuia:
。。。
break;
case adjuz:
。。。
break;
default:
}
return false;
}
3.优化方向:
- 简单可扩展:新增case不需要动原代码逻辑,只需简单操作即可
- 功能内聚:每个case只关注和影响自己
- 可维护: 需要动那个一个渠道,就到对应对象中修改即可
核心代码
3.1 枚举类
public enum CpaSourceEnum {
unknown(1,"未知",""),
guangdiantong(2,"广点通","guangdiantongCallBack"),
wangxiang(3,"旺翔","wangxiangCallBack"),
jinritoutiao(4,"今日头条","jinritoutiaoCallBack"),
tuia(5,"推啊","tuiaCallBack"),
adjuz(6,"巨掌","adjuzCallBack");
CpaSourceEnum(int value, String description, String defaultBeanName) {
this.value = value;
this.description = description;
this.defaultBeanName = defaultBeanName;
}
private int value;
private String description;
// SpringBean,用于找到对应的对象
private String defaultBeanName;
}
3.2 抽象类
抽象策略角色
public abstract class CpaCallBack {
protected Logger logger = LoggerFactory.getLogger(this.getClass());
/**
* 回调广告商
*/
abstract Boolean callbackAdvertisers(CpaUnionInfo cpaUnionInfo);
}
3.3 具体实现
具体策略角色,以今日头条回调为例,其它类似。
@Component
public class JinritoutiaoCallBack extends CpaCallBack {
@Override
public Boolean callbackAdvertisers(CpaUnionInfo cpaUnionInfo) {
。。。
return false;
}
}
3.4 聚合操作类
策略调度与执行者
@Component
public class CompositeCpaCallBack implements ApplicationContextAware {
private Logger logger = LoggerFactory.getLogger(this.getClass());
private ApplicationContext applicationContext;
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.applicationContext=applicationContext;
}
public Boolean callbackAdvertisers(CpaUnionInfo cpaUnionInfo) {
try {
CpaCallBack callBack = getCpaCallBack(cpaUnionInfo.getSource());
return callBack.callbackAdvertisers(cpaUnionInfo);
} catch (Exception e) {
logger.error("回调异常,{}",e);
return false;
}
}
public CpaCallBack getCpaCallBack(String source){
CpaSourceEnum cpaSourceEnum = CpaSourceEnum.valueOf(source);
return (CpaCallBack) applicationContext.getBean(cpaSourceEnum.getDefaultBeanName());
}
}