首页 存档 技术 查看内容

Java程序员从笨鸟到菜鸟之(三十四)大话设计模式(四)策略模式

2018-3-30 13:00 |来自: 互联网 409 0

摘要: 【新朋友】点击标题下面(↑)蓝色字“Java那些事”关注 【老朋友】点击右上角,转发或分享本页面内容 Strategy是属于设计模式中 对象行为型模式,主要是定义一系列的算法,把这些算法一个个封装成单独的类。 定义 ...

【新朋友】点击标题下面(↑)蓝色字“Java那些事”关注

【老朋友】点击右上角,转发或分享本页面内容


Strategy是属于设计模式中 对象行为型模式,主要是定义一系列的算法,把这些算法一个个封装成单独的类。

定义:策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换。策略模式让算法独立于使用它的客户而独立变化。(原文:The Strategy Pattern defines a family ofalgorithms,encapsulates each one,and makes them interchangeable. Strategy letsthe algorithm vary independently from clients that use it.)

角色:

Strategy:策略接口,用来约束一系列具体的策略算法。Context使用这个接口来调用具体的策略实现定义的算法。

ConcreteStrategy:具体的策略实现,也就是具体的算法实现。

Context:上下文,负责和具体的策略类交互,通常上下文会持有一个真正的策略实现,上下文还可以让具体的策略类来获取上下文的数据,甚至让具体的策略类来回调上下文的方法。

策略模式的结构示意图如图1所示:


应用场景:

  1、 多个类只区别在表现行为不同,可以使用Strategy模式,在运行时动态选择具体要执行的行为。

  2、 需要在不同情况下使用不同的策略(算法),或者策略还可能在未来用其它方式来实现。

  3、 对客户隐藏具体策略(算法)的实现细节,彼此完全独立。

优点:

1、 简化了单元测试,因为每个算法都有自己的类,可以通过自己的接口单独测试。
  2、 避免程序中使用多重条件转移语句,使系统更灵活,并易于扩展。
3、 遵守大部分GRASP原则和常用设计原则,高内聚、低偶合。

4.提供了一种替代继承的方法,而且既保持了继承的优点(代码重用)还比继承更灵活(算法独立,可以任意扩展)。

缺点:
  1、 因为每个具体策略类都会产生一个新类,所以会增加系统需要维护的类的数量。
2、 在基本的策略模式中,选择所用具体实现的职责由客户端对象承担,并转给策略模式的Context对象。(这本身没有解除客户端需要选择判断的压力,而策略 模式与简单工厂模式结合后,选择具体实现的职责也可以由Context来承担,这就最大化的减轻了客户端的压力。)

举一个我看过的策略模式比较经典的例子:报价管理

对不同的客户要报不同的价格,向客户报价是非常复杂的,因此在一些CRM(客户关系管理)的系统中,会有一个单独的报价管理模块,来处理复杂的报价功能。

为了演示的简洁性,假定现在需要实现一个简化的报价管理,实现如下的功能:

(1)对普通客户或者是新客户报全价

(2)对老客户报的价格,统一折扣5%

(3)对大客户报的价格,统一折扣10%

(1)先看策略接口,示例代码如下:

[java] view plaincopyprint?

  1. /**

  2. * 策略,定义计算报价算法的接口

  3. */

  4. public inte**ce Strategy {

  5. /**

  6. * 计算应报的价格

  7. * @param goodsPrice 商品销售原价

  8. * @return 计算出来的,应该给客户报的价格

  9. */

  10. public double calcPrice(double goodsPrice);

  11. }



(2)接下来看看具体的算法实现,不同的算法,实现也不一样,先看为新客户或者是普通客户计算应报的价格的实现,示例代码如下:

[java] view plaincopyprint?

  1. /**

  2. * 具体算法实现,为新客户或者是普通客户计算应报的价格

  3. */

  4. public class NormalCustomerStrategy implements Strategy{

  5. public double calcPrice(double goodsPrice) {

  6. System.out.println("对于新客户或者是普通客户,没有折扣");

  7. return goodsPrice;

  8. }

  9. }



再看看为老客户计算应报的价格的实现,示例代码如下:

[java] view plaincopyprint?

  1. /**

  2. * 具体算法实现,为老客户计算应报的价格

  3. */

  4. public class OldCustomerStrategy implements Strategy{

  5. public double calcPrice(double goodsPrice) {

  6. System.out.println("对于老客户,统一折扣5%");

  7. return goodsPrice*(1-0.05);

  8. }

  9. }



再看看为大客户计算应报的价格的实现,示例代码如下:

[java] view plaincopyprint?

  1. /**

  2. * 具体算法实现,为大客户计算应报的价格

  3. */

  4. public class LargeCustomerStrategy implements Strategy{

  5. public double calcPrice(double goodsPrice) {

  6. System.out.println("对于大客户,统一折扣10%");

  7. return goodsPrice*(1-0.1);

  8. }

  9. }



(3)接下来看看上下文的实现,也就是原来的价格类,它的变化比较大,主要有:

  • 原来那些私有的,用来做不同计算的方法,已经去掉了,独立出去做成了算法类

  • 原来报价方法里面,对具体计算方式的判断,去掉了,让客户端来完成选择具体算法的功能

  • 新添加持有一个具体的算法实现,通过构造方法传入

  • 原来报价方法的实现,变化成了转调具体算法来实现

示例代码如下:

[java] view plaincopyprint?

  1. /**

  2. * 价格管理,主要完成计算向客户所报价格的功能

  3. */

  4. public class Price {

  5. /**

  6. * 持有一个具体的策略对象

  7. */

  8. private Strategy strategy = null;

  9. /**

  10. * 构造方法,传入一个具体的策略对象

  11. * @param aStrategy 具体的策略对象

  12. */

  13. public Price(Strategy aStrategy){

  14. this.strategy = aStrategy;

  15. }

  16. /**

  17. * 报价,计算对客户的报价

  18. * @param goodsPrice 商品销售原价

  19. * @return 计算出来的,应该给客户报的价格

  20. */

  21. public double quote(double goodsPrice){

  22. return this.strategy.calcPrice(goodsPrice);

  23. }

  24. }



(4)写个客户端来测试运行一下,好加深体会,示例代码如下:

[html] view plaincopyprint?

  1. public class Client {

  2. public static void main(String[] args) {

  3. //1:选择并创建需要使用的策略对象

  4. Strategy strategy = new LargeCustomerStrategy ();

  5. //2:创建上下文

  6. Price ctx = new Price(strategy);

  7. //3:计算报价

  8. double quote = ctx.quote(1000);

  9. System.out.println("向客户报价:" quote);

  10. }

  11. }


本文来自:曹胜欢博客专栏。转载请注明出处:http://blog.csdn.net/csh624366188


下一篇写:

Java程序员从笨鸟到菜鸟之(三十四)大话设计模式(五)创建者模式和原型模式


本文转载自:微信公众账号 - java那些事,版权归原作者所有!

声明:文章版权归原作者所有 部分文章转自互联网 如有侵权请联系 [邮箱地址] 删除

路过

雷人

握手

鲜花

鸡蛋

相关分类

返回顶部