策略模式对应于解决某一个问题的算法族,允许用户从该算法族中任选一个算法解决某一问题,同时可以方便的更换算法或者增加新的算法。并且由客户端决定调用哪个算法。
场景
- 某个市场人员接到单后的报价策略(CRM系统中常见问题)。报价策略很复杂,简单做如下分类:
- 普通客户小批量报价
- 普通客户大批量报价
- 老客户小批量报价
- 老客户大批量报价
- 具体选用哪个报价策略,这需要根据实际情况来确定。
我们可以先用 条件语句(if) 处理,实现起来比较容易,符合一般开发人员的思路。
但假如条件特别多,算法比较复杂时,整个条件代码会变得很长,难以维护。如有新增类型,就需要频繁的修改此处的代码。这时候,可以采用策略模式。
案例
/**
* @author huangyzh
* @create 2020-05-20 0:20
*/
public interface Strategy {
double getPrice(double standardPrice);
}
/**
* @author huangyzh
* @create 2020-05-20 0:21
*/
public class NewCustomerFewStrategy implements Strategy {
@Override
public double getPrice(double standardPrice) {
System.out.println("普通客户小批量……不打折,原价");
return standardPrice;
}
}
/**
* @author huangyzh
* @create 2020-05-20 0:22
*/
public class NewCustomerManyStrategy implements Strategy {
@Override
public double getPrice(double standardPrice) {
System.out.println("普通客户大批量……打九折!");
return standardPrice*0.9;
}
}
/**
* @author huangyzh
* @create 2020-05-20 0:21
*/
public class OldCustomerFewStrategy implements Strategy {
@Override
public double getPrice(double standardPrice) {
System.out.println("老客户小批量……85折");
return standardPrice*0.85;
}
}
/**
* @author huangyzh
* @create 2020-05-20 0:23
*/
public class OldCustomerManyStrategy implements Strategy {
@Override
public double getPrice(double standardPrice) {
System.out.println("老客户大批量……7折");
return standardPrice*0.7;
}
}
/**
* 负责和具体的策略类交互
* 这样就可以将具体的算法和直接的调用方分离,使得算法可以独立于调用方
* @author huangyzh
* @create 2020-05-20 0:24
*/
public class Context {
private Strategy strategy;//当前采用的算法对象
public Context(Strategy strategy) {
this.strategy = strategy;
}
public void printPrice(double s){
System.out.println("报价:"+strategy.getPrice(s));
}
}
/**
* @author huangyzh
* @create 2020-05-20 0:27
*/
public class Test01 {
public static void main(String[] args) {
Strategy strategy = new OldCustomerManyStrategy();
Context c = new Context(strategy);
c.printPrice(359.0);
}
}
本质
分离算法,选择实现
开发常见场景
- JavaSE中GUI编程中,布局管理
- Spring框架中,Resource接口,资源访问策略
- javax.servlet.http.HttpServlet#service()