在工作中策略模式可以说是应用场景最多的一种设计模式了, 但大多数人却还停留在概念层, 不能实际应用. 本文就一起看下如何使用策略模式.
1
策略模式
策略模式是对算法的包装, 并将每个算法单独封装起来, 且能在一定条件下进行互换.
共有三个角色:
调用上下文(Context): 持有一个Strategy的引用.
抽象策略或算法(Strategy): 这是一个抽象角色, 通常由一个接口或抽象类实现, 并给出所需的接口的定义或方法声明.
具体策略或算法(ConcreteStrategy): 包装了相关的算法或行为.
2
举个例子
当需要实现一个可以选择汽车,火车等多出行方式的逻辑时, 大概你会写成这样.
代码语言:javascript复制if(type.equals("CAR")){
// do something
} else if(type.equals("TRAIN")){
// do something
}
如果某天又要加一个新出行方式, 就需要找到这代码, 再添加if语句, 还有可能因为依赖上下文的不同, 影响到原有的逻辑. 这时候就该策略模式登场了.
通过定义getType()方法, 确定适用场景.
代码语言:javascript复制interface Transportation {
String getType();
void doSomething();
}
class Car implements Transportation {
@Override
public String getType() {
return "CAR";
}
@Override
public void doSomething() {
// do something
}
}
class Train implements Transportation {
@Override
public String getType() {
return "TRAIN";
}
@Override
public void doSomething() {
// do something
}
}
将行为抽象出来, 在有新的出行方式时, 增加新类就可以了, 对原有逻辑没有任何影响.
3
调用端
可以发现只需要做好算法集合初始化即可, 有新算法添加到集合中即可.
代码语言:javascript复制public static void main(String[] args) {
Transportation car = new Car();
Transportation train = new Train();
List<Transportation> list = Lists.newArrayList(car, train);
String type = "";
for (int i = 0; i < list.size(); i ) {
Transportation transportation = list.get(i);
if(type.equals(transportation.getType())){
transportation.doSomething();
}
}
}
小结
策略模式的重心不是如何实现算法,而是如何组织调用这些算法,从而让程序结构更灵活,具有更好的维护性和扩展性. 也就是常说的