java种设计模式: 策略模式

策略模式属于对象的行为模式。其用意是针对一组算法,将每一个算法封装到具有共同接口的独立的类中,从而使得它们可以相互替换。策略模式使得算法可以在不影响到客户端的情况下发生变化。

  策略模式把一个系列的算法封装到一个系列的具体策略类里面,作为一个抽象策略类的子类或策略接口的实现类。简单地说:准备一组算法,并将每一个算法封装起来,使它们可以互换。

  

  示意性UML图:

  

  这个模式涉及到3个角色:

  环境(Context)角色:持有一个Strategy抽象策略类或策略接口的引用。

  抽象策略(Strategy)角色:这是一个抽象角色,由一个接口或抽象类实现。此角色声明所有的具体策略类需要重写的方法。

  具体策略(Concrete Strategy)角色:封装了相关的算法或行为。

  示意性实例:

  环境角色

复制代码
 1 public class Context {
 2     // 持有一个具体策略的对象
 3     private Strategy strategy;
 4     /**
 5      * 构造函数,传入一个具体策略对象
 6      * @param strategy    具体策略对象
 7      */
 8     public Context(Strategy strategy){
 9         this.strategy = strategy;
10     }
11     /**
12      * 策略方法
13      */
14     public void contextInterface(){
15         
16         strategy.strategyInterface();
17     }
18     
19 }
复制代码

  抽象策略角色

复制代码
1 public interface Strategy {
2     /**
3      * 策略方法
4      */
5     public void strategyInterface();
6 }
复制代码

  具体策略角色

复制代码
1 public class ConcreteStrategyA implements Strategy {
2 
3     @Override
4     public void strategyInterface() {
5         // 相关的业务
6     }
7 
8 }
复制代码

  

复制代码
1 public class ConcreteStrategyB implements Strategy {
2 
3     @Override
4     public void strategyInterface() {
5         // 相关的业务
6     }
7 
8 }
复制代码

 

复制代码
1 public class ConcreteStrategyC implements Strategy {
2 
3     @Override
4     public void strategyInterface() {
5         // 相关的业务
6     }
7 
8 }
复制代码

 

  使用场景实例:

  假设某个网站销售各种书籍,对初级会员没有提供折扣,对中级会员提供每本10%的促销折扣,对高级会员提供每本20%的促销折扣。

  折扣是根据以下的3个算法中的1个进行的:

  算法1:对初级会员没有提供折扣。

  算法2:对中级会员提供10%的促销折扣。

  算法3:对高级会员提供20%的促销折扣。

  该实例的UML图:

  

  折扣接口

复制代码
1 public interface MemberStrategy {
2     /**
3      * 计算图书的价格
4      * @param booksPrice    图书的原价
5      * @return    计算出打折后的价格
6      */
7     public double calcPrice(double booksPrice);
8 }
复制代码

  初级会员折扣实现类

复制代码
 1 public class PrimaryMemberStrategy implements MemberStrategy {
 2 
 3     @Override
 4     public double calcPrice(double booksPrice) {
 5         
 6         System.out.println("对于初级会员的没有折扣");
 7         return booksPrice;
 8     }
 9 
10 }
复制代码

  中级会员折扣实现类

 

复制代码
 1 public class IntermediateMemberStrategy implements MemberStrategy {
 2 
 3     @Override
 4     public double calcPrice(double booksPrice) {
 5 
 6         System.out.println("对于中级会员的折扣为10%");
 7         return booksPrice * 0.9;
 8     }
 9 
10 }
复制代码

 

  高级会员折扣实现类

 

复制代码
1 public class AdvancedMemberStrategy implements MemberStrategy {
2 
3     @Override
4     public double calcPrice(double booksPrice) {
5         
6         System.out.println("对于高级会员的折扣为20%");
7         return booksPrice * 0.8;
8     }
9 }
复制代码

 

  价格类

 

复制代码
 1 public class Price {
 2     // 持有一个具体的策略对象
 3     private MemberStrategy strategy;
 4     /**
 5      * 构造函数,传入一个具体的策略对象
 6      * @param strategy    具体的策略对象
 7      */
 8     public Price(MemberStrategy strategy){
 9         this.strategy = strategy;
10     }
11     
12     /**
13      * 计算图书的价格
14      * @param booksPrice    图书的原价
15      * @return    计算出打折后的价格
16      */
17     public double quote(double booksPrice){
18         return this.strategy.calcPrice(booksPrice);
19     }
20 }
复制代码

 

  客户端

 

复制代码
 1 public class Client {
 2 
 3     public static void main(String[] args) {
 4         // 选择并创建需要使用的策略对象
 5         MemberStrategy strategy = new AdvancedMemberStrategy();
 6         // 创建环境
 7         Price price = new Price(strategy);
 8         // 计算价格
 9         double quote = price.quote(300);
10         System.out.println("图书的最终价格为:" + quote);
11     }
12 
13 }
复制代码

 

 

  策略模式的重心不是如何实现算法,而是如何组织、调用这些算法,从而让程序结构更灵活,具有更好的维护性和扩展性。策略算法是相同行为的不同实现。在运行期间,策略模式在每一个时刻只能使用一个具体的策略实现对象。把所有的具体策略实现类的共同公有方法封装到抽象类里面,将代码向继承等级结构的上方集中。

  

  

  策略模式优点:

  1 通过策略类的等级结构来管理算法族。

  2 避免使用将采用哪个算法的选择与算法本身的实现混合在一起的多重条件(if-else if-else)语句。

  策略模式缺点:

  1 客户端必须知道所有的策略类,并自行决定使用哪一个策略类。

  2 由于策略模式把每个算法的具体实现都单独封装成类,针对不同的情况生成的对象就会变得很多。

 

  参考资料

  《JAVA与模式》之策略模式


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值