策略模式是一种设计模式,它定义了一系列的算法,并将每一个算法封装起来,使它们可以互相替换。策略模式使得算法可以独立于使用它的客户而变化,从而提高了代码的灵活性和可维护性。
策略模式通常用于解决多种相关类仅仅是行为有异的问题,或者需要使用一个算法的不同变体的情况。通过使用策略模式,我们可以将算法的使用与算法的实现分离开来,使得算法的变化不会影响到使用它的代码。
在策略模式中,通常有一个抽象策略角色,它是一个接口或抽象类,定义了算法的公共接口。然后,有一系列的策略类实现了这个接口或抽象类,每个策略类代表一种具体的算法实现。这些策略类可以独立变化,而不会影响到使用它们的代码。
策略模式在实际工作中算是登场频率较高的设计模式,例如供应链业务下业务是按照租户划分的,这时候如果搞优惠活动各租户的优惠力度或者模式可能不同,这时候就可以用策略模式优雅的完成不同租户下优惠方式的选择。
以下是一个表示学习和工作的策略demo:
context类(Context类在策略模式中起到了连接客户端和策略对象、封装策略选择和使用过程、提供灵活策略切换机制以及简化客户端调用的作用):
public class Context {
//获取策略路由
Strategy strategy;
public Strategy getStrategy() {
return strategy;
}
public void setStrategy(Strategy strategy) {
this.strategy = strategy;
}
//执行策略
void contextStrategy() {
strategy.execute();
}
}
strategy接口(定义策略方法,一般有获取策略和执行策略方法,具体方法由对应策略实现类去实现):
public interface Strategy {
//获取策略
String getStrategy();
//执行策略
void execute();
}
两个策略实现类,分别代表学习和工作:
public class StudyStrategy implements Strategy{
@Override
public String getStrategy() {
return "study...";
}
@Override
public void execute() {
System.out.println("I want to study...");
}
}
public class WorkStrategy implements Strategy{
@Override
public String getStrategy() {
return "work...";
}
@Override
public void execute() {
System.out.println("I want to work...");
}
}
StrategyFactory策略工厂类,用于管理策略,以及控制策略的选择:
public class StrategyFactory {
//通过策略工厂选择使用哪种策略
public Strategy getStrategyByFactory(String strategy) {
if ("study".equals(strategy)) {
return new StudyStrategy();
} else if ("work".equals(strategy)) {
return new WorkStrategy();
} else {
throw new IllegalArgumentException("I do not want to do other things...");
}
}
}
创建一个客户端调用:
public class Execute {
public static void execute(){
Context context = new Context();
//如果不用策略工厂进行管理创建策略,那么直接在上下文中设置对应策略
/*context.setStrategy(new StudyStrategy());*/
//用策略工厂进行管理策略,直接输入策略的标识,创建的过程就交给策略工厂
StrategyFactory strategyFactory = new StrategyFactory();
context.setStrategy(strategyFactory.getStrategyByFactory("study"));
System.out.println(context.getStrategy().getStrategy());
context.contextStrategy();
}
public static void main(String[] args) {
execute();
}
}
输出: