`

23种设计模式(12):策略模式

 
阅读更多

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

 

策略模式的结构

     策略模式是对算法的包装,是把使用算法的责任和算法本身分割开来,委派给不同的对象管理。策略模式通常把一个系列的算法包装到一系列的策略类里面,作为一个抽象策略类的子类。用一句话来说,就是:“准备一组算法,并将每一个算法封装起来,使得它们可以互换”。下面就以一个示意性的实现讲解策略模式实例的结构。

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

  ●  环境(Context)角色:持有一个Strategy的引用。

  ●  抽象策略(Strategy)角色:这是一个抽象角色,通常由一个接口或抽象类实现。此角色给出所有的具体策略类所需的接口。

  ●  具体策略(ConcreteStrategy)角色:包装了相关的算法或行为。

源代码

  环境角色类

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  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. }  

抽象策略类
[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. public interface Strategy {  
  2.     /** 
  3.      * 策略方法 
  4.      */  
  5.     public void strategyInterface();  
  6. }  
具体策略类
[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. public class ConcreteStrategyA implements Strategy {  
  2.   
  3.     @Override  
  4.     public void strategyInterface() {  
  5.         //相关的业务  
  6.     }  
  7.   
  8. }  
[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. public class ConcreteStrategyB implements Strategy {  
  2.   
  3.     @Override  
  4.     public void strategyInterface() {  
  5.         //相关的业务  
  6.     }  
  7.   
  8. }  

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. public class ConcreteStrategyC implements Strategy {  
  2.   
  3.     @Override  
  4.     public void strategyInterface() {  
  5.         //相关的业务  
  6.     }  
  7.   
  8. }  
 

使用场景

  假设现在要设计一个贩卖各类书籍的电子商务网站的购物车系统。一个最简单的情况就是把所有货品的单价乘上数量,但是实际情况肯定比这要复杂。比如,本网站可能对所有的高级会员提供每本20%的促销折扣;对中级会员提供每本10%的促销折扣;对初级会员没有折扣。

  根据描述,折扣是根据以下的几个算法中的一个进行的:

  算法一:对初级会员没有折扣。

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

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

  使用策略模式来实现的结构图如下:





 

源代码

 

  抽象折扣类

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. public interface MemberStrategy {  
  2.     /** 
  3.      * 计算图书的价格 
  4.      * @param booksPrice    图书的原价 
  5.      * @return    计算出打折后的价格 
  6.      */  
  7.     public double calcPrice(double booksPrice);  
  8. }  

初级会员折扣类

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  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. }  

中级会员折扣类

 

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  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. }  

高级会员折扣类

 

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  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. }  

价格类

 

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  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. }  

客户端

 

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  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)语句。多重条件语句不易维护,它把采取哪一种算法或采取哪一种行为的逻辑与算法或行为的逻辑混合在一起,统统列在一个多重条件语句里面,比使用继承的办法还要原始和落后。

策略模式的缺点

  (1)客户端必须知道所有的策略类,并自行决定使用哪一个策略类。这就意味着客户端必须理解这些算法的区别,以便适时选择恰当的算法类。换言之,策略模式只适用于客户端知道算法或行为的情况。

  (2)由于策略模式把每个具体的策略实现都单独封装成为类,如果备选的策略很多的话,那么对象的数目就会很可观。


原文转载于 http://www.cnblogs.com/java-my-life/archive/2012/05/10/2491891.html

分享到:
评论

相关推荐

    23种设计模式之:建造者,代理,观察,策略,状态模式

    本文将重点介绍标题中提到的五种设计模式:建造者模式、代理模式、观察者模式、策略模式以及状态模式。 **1. 建造者模式(Builder)** 建造者模式是一种创建型设计模式,它允许我们分步骤构建复杂对象,而无需暴露...

    23种设计模式详解PDF

    设计模式 的分类 总体来说设计模式分为三大类: 创建型模式(5): ...策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。

    GOF23种设计模式

    GOF(GoF)23种设计模式,是由Erich Gamma、Richard Helm、Ralph Johnson和John Vlissides四位大神在他们的著作《设计模式:可复用面向对象软件的基础》中提出的,这些模式分为创建型、结构型和行为型三大类。...

    23种java设计模式.pdf

    JAVA 设计模式可以分为三种:创建模式、结构模式和行为模式。 1. 创建模式 创建模式是指在创建对象时使用的模式,包括 Factory(工厂模式)、Singleton(单例模式)、Builder(建造者模式)、Prototype(原型模式...

    java 23种设计模式.zip

    设计模式主要分为三大类: 1.创建型模式:工厂模式、抽象工厂模式、单例模式、建造者模式、原型模式。 2.结构型模式:适配器模式、桥接模式、装饰模式、组合模式、外观模式、享元模式、代理模式。 4.行为型模式:...

    java23种设计模式详解+源码(绝对详解)

    在Java编程中,有23种经典的GoF(Gang of Four)设计模式,它们被分为三大类:创建型、结构型和行为型。本资源集合了这些模式的详细解释与源码分析,旨在帮助开发者深入理解和应用设计模式。 1. 创建型模式...

    23种设计模式demo

    java的设计模式大体上分为三大类: ...行为型模式(11种):策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。

    Java之23种设计模式解析

    本资源“Java之23种设计模式解析”源自尚硅谷教育机构,由宋红康老师主讲的“玩转Java”系列课程。这份资料详细介绍了在Java编程中常用的23种设计模式,旨在提升开发者对于软件设计的理解和应用能力,从而写出更加...

    23种 设计模式

    在《设计模式》课件中,详细讲解了23种经典的GOF(GoF,Gamma, Helm, Johnson, Vlissides)设计模式,这些模式分为三大类:创建型、结构型和行为型。 1. 创建型设计模式: - 单例模式:确保一个类只有一个实例,并...

    C#23种设计模式【完整】.pdf

    本文档详细介绍了23种C#设计模式,包括创建型、结构型和行为型三个大类。这些设计模式是.NET进阶必备的知识,通过学习和掌握这些设计模式,可以提高程序员的设计和编码能力。 创建型设计模式 1. 单件模式...

    23种设计模式(C++).pdf

    《设计模式精解-GoF 23 种设计模式解析附 C++实现源码》是一本深入探讨软件设计模式的书籍,它涵盖了创建型、结构型和行为型三种主要类型的23个经典设计模式,并提供了C++语言的实现代码。设计模式是软件工程中的...

    设计模式 23种设计模式PPT

    设计模式是软件开发中一种广泛采用的实践,它代表了在特定上下文中解决常见问题的通用解决方案。设计模式并非具体的代码或库,而是对最佳实践的描述,它们是经过时间验证、可重用的代码设计模式,旨在提高代码的...

    java之23种设计模式完整代码

    这里我们探讨的“Java之23种设计模式完整代码”是一份宝贵的资源,它包含了所有23种经典设计模式的实现示例。这份资料能够帮助开发者深入理解每种设计模式的概念、应用场景以及它们之间的差异。 设计模式是经过时间...

    23种面向对象设计模式

    标题提到的“23种面向对象设计模式”涵盖了设计模式的主要分类,这些模式在Java、C++等面向对象编程语言中广泛应用。 1. **创建型模式**(Creational Patterns):这类模式关注对象的创建过程,包括单例模式...

    Java 经典设计模式讲解以及项目实战

    Java 经典设计模式讲解以及项目实战 设计模式简介:主要介绍各种设计模式的概念和运用场景等 设计模式综合运用:主要是笔者在实际工作中运用到的一些设计模式综合运用事例的提炼 Spring设计模式简介:主要是讲述...

    Java23种设计模式可直接运行Demo

    行为型模式,共十一种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。 其实还有两类:并发型模式和线程池模式。

    设计模式精解-GoF-23种设计模式解析--附C++源代码

    这本书详细阐述了23种设计模式,这些模式被广泛应用于各种编程语言,包括C++。 1. **创建型模式**:这类模式主要关注对象的创建过程,它们提供了一种在不指定具体类的情况下创建对象的方法,使得系统更加灵活和可...

    java23种设计模式

    java23种设计模式,每一种模式都有详细的讲解,很全面,如果你想深入了解一下java23种设计模式,这会非常适合你的哦!上传乃为百度云连接,失效请留言。 内容: 001策略模式! N. B8 ~' D! f9 j+ g0 I 002观察者模式 ...

    设计模式精解-GoF23种设计模式解析附C++实现源码

    GoF(Gang of Four)23种设计模式是软件开发中的经典,由Erich Gamma、Richard Helm、Ralph Johnson和John Vlissides四位专家在《设计模式:可复用面向对象软件的基础》一书中提出。这些模式为程序员提供了一种通用...

    java常用23中设计模式

    行为型模式,共十一种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。 其实还有两类:并发型模式和线程池模式。

Global site tag (gtag.js) - Google Analytics