`

策略模式

阅读更多
本文探讨初学使用策略模式时遇到的一些疑惑,以及在工作中慢慢解决之前遇到的疑惑,借此与大家分享。比如说本文谈到策略模式中环境角色Context的用处,为什么一定要用,可不可以将此取消。这些都是在学习和工作的实践总结中慢慢体会到的。
首先,我们来看下策略模式的概念。一般的解释如下: 
策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换。策略模式让算法独立于使用它的客户而独立变化。(原文:The Strategy Pattern defines a family of algorithms,encapsulates each one,and makes them interchangeable. Strategy lets the algorithm vary independently from clients that use it.)

一般的,策略模式主要分为以下三个角色:
1. 环境角色(Context):持有一个策略类引用
2. 抽象策略(Strategy):定义了多个具体策略的公共接口,具体策略类中各种不同的算法以不同的方式实现这个接口;Context使用这些接口调用不同实现的算法。一般的,我们使用接口或抽象类实现。
3. 具体策略(ConcreteStrategy):实现抽象策略类中的相关的算法或操作。

我们首先来写一个简单的策略模式,然后再结合实际应用进行扩展,进而思考其在实际开发中的使用方法。
(给大家推荐前辈写的一篇不错的博文,研磨设计模式之 策略模式http://www.uml.org.cn/sjms/201009092.asp
我们这个简单的策略假设就是想让不同的策略来实现某个算法(algorithm),抽象类如下:
Java代码 复制代码 收藏代码
  1. package com.icecode.demo.strategy;
  2. /**
  3. * 抽象策略
  4. * @author zhanche
  5. *
  6. */
  7. public abstract class AbstractStrategy {
  8. /**
  9. * 某个希望有不同策略实现的算法
  10. */
  11. public abstract void algorithm();
  12. }

算法algorithm的具体实现策略类ConcreteStrategy1和ConcreteStrategy2如下:
Java代码 复制代码 收藏代码
  1. package com.icecode.demo.strategy.impl;
  2. import com.icecode.demo.strategy.AbstractStrategy;
  3. /**
  4. * 对算法的第一种具体实现策略
  5. * @author zhanche
  6. *
  7. */
  8. public class ConcreteStrategy1 extends AbstractStrategy {
  9. @Override
  10. public void algorithm() {
  11. System.out.println("----------------我是策略一算法----------------");
  12. }
  13. }
  14. package com.icecode.demo.strategy.impl;
  15. import com.icecode.demo.strategy.AbstractStrategy;
  16. /**
  17. * 对算法的第二种具体实现策略
  18. * @author zhanche
  19. *
  20. */
  21. public class ConcreteStrategy2 extends AbstractStrategy {
  22. @Override
  23. public void algorithm() {
  24. System.out.println("----------------我是策略二算法----------------");
  25. }
  26. }


环境角色的实现如下:
Java代码 复制代码 收藏代码
  1. package com.icecode.demo.context;
  2. import com.icecode.demo.strategy.AbstractStrategy;
  3. /**
  4. * 环境角色,主要完成对特定策略的调用
  5. * @author zhanche
  6. *
  7. */
  8. public class Context {
  9. private AbstractStrategy strategy;
  10. public Context(AbstractStrategy strategy) {
  11. this.strategy = strategy;
  12. }
  13. public void algorithm() {
  14. this.strategy.algorithm();
  15. }
  16. }


下面简单写一个客户端测试的代码:
Java代码 复制代码 收藏代码
  1. package com.icecode.demo;
  2. import com.icecode.demo.context.Context;
  3. import com.icecode.demo.strategy.impl.ConcreteStrategy1;
  4. import com.icecode.demo.strategy.impl.ConcreteStrategy2;
  5. /**
  6. * 策略模式测试类
  7. * @author zhanche
  8. *
  9. */
  10. public class Client {
  11. /**
  12. * @param args
  13. */
  14. public static void main(String[] args) {
  15. Context context = new Context(new ConcreteStrategy1());
  16. context.algorithm();
  17. context = new Context(new ConcreteStrategy2());
  18. context.algorithm();
  19. }
  20. }


输出结果如下:
----------------我是策略一算法----------------
----------------我是策略二算法----------------



好吧,到此为止,一个简单的策略模式就写完了。但是,大家肯定有所疑惑,Context完全可以没有嘛,既然抽象策略AbstractStrategy已经持有algorithm这个接口,我们完全可以如下去写代码,让系统根据不同的实现执行不同的策略不就完了。代码可以如下:
Java代码 复制代码 收藏代码
  1. package com.icecode.demo;
  2. import com.icecode.demo.strategy.AbstractStrategy;
  3. import com.icecode.demo.strategy.impl.ConcreteStrategy1;
  4. import com.icecode.demo.strategy.impl.ConcreteStrategy2;
  5. /**
  6. * 策略模式测试类
  7. * @author zhanche
  8. *
  9. */
  10. public class Client {
  11. /**
  12. * @param args
  13. */
  14. public static void main(String[] args) {
  15. AbstractStrategy strategy = new ConcreteStrategy1();
  16. strategy.algorithm();
  17. strategy = new ConcreteStrategy2();
  18. strategy.algorithm();
  19. }
  20. }

输出结果如下:
----------------我是策略一算法----------------
----------------我是策略二算法----------------


可见,2种方案都实现了同一个引用根据不同的实现执行特定的算法。是的,分析发现,在上面简单的应用中,Context的确可以取消。那么,Context这个角色为什么还要存在呢?
让我们考虑以下几种情况:
1、如果我们需要对不同策略中相同算法的参数,执行相同的安全性检查,我们如果没有环境角色Context,则只能在每个实现的开始部分,调用安全性检查代码;而有了Context这个角色,我们可以在调用Context的构造器时,统一进行安全性检查。这在我们的实现策略比较多的时候,比如说7、8个的时候,特别有用,可以大量减少冗余的代码量。
2、如果我们需要改变原有算法时,需要引进新的参数,如果没有Context,我们怎么办?一种办法是重载该算法,增加新的函数接口;另外一种办法是完全废弃原有的函数接口,重新写新的函数接口。毋庸置疑,这2种办法的代价都很大,尤其是如果这个新的参数只有部分实现策略中的该算法实现用到的时候。而我们使用Context就可以完全解决这个问题。
下面我们改造下上面那个基本策略模式,我们让策略模式也持有对Context的引用,这样的优点是可以在策略类里回调的Context里的所有可用的变量或函数等信息。此外,我们也增加一个新的实现策略类ConcreteStrategy3,具体代码如下所示:
Java代码 复制代码 收藏代码
  1. package com.icecode.demo.strategy;
  2. import com.icecode.demo.context.Context;
  3. /**
  4. * 抽象策略
  5. * @author zhanche
  6. *
  7. */
  8. public abstract class AbstractStrategy {
  9. /**
  10. * 某个希望有不同策略实现的算法
  11. */
  12. public abstract void algorithm(Context context);
  13. }
  14. package com.icecode.demo.strategy.impl;
  15. import com.icecode.demo.context.Context;
  16. import com.icecode.demo.strategy.AbstractStrategy;
  17. /**
  18. * 对算法的第一种具体实现策略
  19. * @author zhanche
  20. *
  21. */
  22. public class ConcreteStrategy1 extends AbstractStrategy {
  23. @Override
  24. public void algorithm(Context context) {
  25. System.out.println("----------------我是策略一算法----------------");
  26. }
  27. }
  28. package com.icecode.demo.strategy.impl;
  29. import com.icecode.demo.context.Context;
  30. import com.icecode.demo.strategy.AbstractStrategy;
  31. /**
  32. * 对算法的第二种具体实现策略
  33. * @author zhanche
  34. *
  35. */
  36. public class ConcreteStrategy2 extends AbstractStrategy {
  37. @Override
  38. public void algorithm(Context context) {
  39. System.out.println("----------------我是策略二算法----------------");
  40. }
  41. }
  42. package com.icecode.demo.strategy.impl;
  43. import com.icecode.demo.context.Context;
  44. import com.icecode.demo.strategy.AbstractStrategy;
  45. /**
  46. * 对算法的第三种具体实现策略
  47. * @author zhanche
  48. *
  49. */
  50. public class ConcreteStrategy3 extends AbstractStrategy {
  51. @Override
  52. public void algorithm(Context context) {
  53. System.out.println("----------------我是策略三算法----------------");
  54. }
  55. }
  56. package com.icecode.demo.context;
  57. import com.icecode.demo.strategy.AbstractStrategy;
  58. /**
  59. * 环境角色,主要完成对特定策略的调用
  60. * @author zhanche
  61. *
  62. */
  63. public class Context {
  64. /**
  65. * 持有对策略的引用
  66. */
  67. private AbstractStrategy strategy;
  68. /**
  69. * 算法入口
  70. */
  71. public void algorithm() {
  72. this.strategy.algorithm(this);
  73. }
  74. }


好了,现在我们想这样改变需求,ConcreteStrategy2和ConcreteStrategy3里对algorithm算法的实现,需要统计两个新的信息,分别用parameter1和parameter2来表示,同时要统计所有实现策略类里,对algorithm算法调用的次数。
如果没有Context这个角色,又需要做到客户端调用的时候代码改动尽量少,相信大家的做法只好是改抽象策略类和实现策略类里的algorithm算法。但是这样实现策略ConcreteStrategy1可能不愿意了,因为他并不需要新增加的参数;此外,对所有实现类里algorithm算法调用的统计也没有一个统一的入口,需要在每个algorithm实现中,插入一个计数代码。但是如果有了环境角色Context,一切就变得很简单了,我们不需要改动抽象策略类,和实现策略类ConcreteStrategy1,只需要改需求发生变化相关的类,且看下面的代码:
Java代码 复制代码 收藏代码
  1. package com.icecode.demo.context;
  2. import com.icecode.demo.strategy.AbstractStrategy;
  3. /**
  4. * 环境角色,主要完成对特定策略的调用
  5. * @author zhanche
  6. *
  7. */
  8. public class Context {
  9. /**
  10. * 持有对策略的引用
  11. */
  12. private AbstractStrategy strategy;
  13. /**
  14. * parameter1、parameter2只是ConcreteStrategy2ConcreteStrategy3需要使用的参数,
  15. * 而ConcreteStrategy1不使用
  16. */
  17. private int parameter1;
  18. private int parameter2;
  19. //count用来统计所有策略的算法algorithm调用的总次数
  20. public static int count = 0;
  21. public Context(AbstractStrategy strategy) {
  22. this.strategy = strategy;
  23. }
  24. public Context(AbstractStrategy strategy, int parameter1, int parameter2) {
  25. super();
  26. this.strategy = strategy;
  27. this.parameter1 = parameter1;
  28. this.parameter2 = parameter2;
  29. }
  30. public int getParameter1() {
  31. return parameter1;
  32. }
  33. public int getParameter2() {
  34. return parameter2;
  35. }
  36. /**
  37. * 算法入口
  38. */
  39. public void algorithm() {
  40. count++;
  41. System.out.println("------------这是第"+count+"次调用algorithm算法--------");
  42. this.strategy.algorithm(this);
  43. }
  44. }
  45. package com.icecode.demo.strategy.impl;
  46. import com.icecode.demo.context.Context;
  47. import com.icecode.demo.strategy.AbstractStrategy;
  48. /**
  49. * 对算法的第一种具体实现策略
  50. * @author zhanche
  51. *
  52. */
  53. public class ConcreteStrategy1 extends AbstractStrategy {
  54. @Override
  55. public void algorithm(Context context) {
  56. System.out.println("----------------我是策略一算法----------------");
  57. }
  58. }
  59. package com.icecode.demo.strategy.impl;
  60. import com.icecode.demo.context.Context;
  61. import com.icecode.demo.strategy.AbstractStrategy;
  62. /**
  63. * 对算法的第二种具体实现策略
  64. * @author zhanche
  65. *
  66. */
  67. public class ConcreteStrategy2 extends AbstractStrategy {
  68. @Override
  69. public void algorithm(Context context) {
  70. System.out.println("----------------我是策略二算法----------------");
  71. System.out.println("------------------我需要的参数parameter1="+context.getParameter1());
  72. System.out.println("------------------我需要的参数parameter2="+context.getParameter2());
  73. }
  74. }
  75. package com.icecode.demo.strategy.impl;
  76. import com.icecode.demo.context.Context;
  77. import com.icecode.demo.strategy.AbstractStrategy;
  78. /**
  79. * 对算法的第三种具体实现策略
  80. * @author zhanche
  81. *
  82. */
  83. public class ConcreteStrategy3 extends AbstractStrategy {
  84. @Override
  85. public void algorithm(Context context) {
  86. System.out.println("----------------我是策略二算法----------------");
  87. System.out.println("------------------我需要的参数parameter1="+context.getParameter1());
  88. System.out.println("------------------我需要的参数parameter2="+context.getParameter2());
  89. }
  90. }


客户端测试的代码如下:
Java代码 复制代码 收藏代码
  1. package com.icecode.demo;
  2. import com.icecode.demo.context.Context;
  3. import com.icecode.demo.strategy.impl.ConcreteStrategy1;
  4. import com.icecode.demo.strategy.impl.ConcreteStrategy2;
  5. import com.icecode.demo.strategy.impl.ConcreteStrategy3;
  6. /**
  7. * 策略模式测试类
  8. * @author zhanche
  9. *
  10. */
  11. public class Client {
  12. /**
  13. * @param args
  14. */
  15. public static void main(String[] args) {
  16. Context context = new Context(new ConcreteStrategy1());
  17. context.algorithm();
  18. context = new Context(new ConcreteStrategy2(),100, 200);
  19. context.algorithm();
  20. context = new Context(new ConcreteStrategy3(), 100, 200);
  21. context.algorithm();
  22. }
  23. }

测试输出结果如下:
------------这是第1次调用algorithm算法--------
----------------我是策略一算法----------------
------------这是第2次调用algorithm算法--------
----------------我是策略二算法----------------
------------------我需要的参数parameter1=100
------------------我需要的参数parameter2=200
------------这是第3次调用algorithm算法--------
----------------我是策略三算法----------------
------------------我需要的参数parameter1=100
------------------我需要的参数parameter2=200


由以上分析可见,策略模式中,各个角色的功能都非常重要,虽然环境角色Context可以在某些简单的策略模式中不去使用,但是如果无法预测到各个实现策略功能和需求的变化,以及实现灵活性更好的策略模式,在使用策略模式进行架构时,一定要充分利用所有角色的功能。 
分享到:
评论
发表评论

文章已被作者锁定,不允许评论。

相关推荐

    策略模式结合模板方法模式

    策略模式结合模板方法模式的设计思路 策略模式结合模板方法模式是策略模式的一种变形,目的是为了解决策略模式中的一些共性问题。在策略模式中,经常会出现这样一种情况,就是发现这一系列算法的实现上存在公共功能...

    详解SpringBoot结合策略模式实战套路

    SpringBoot结合策略模式实战套路 策略模式是一种常用的设计模式,它可以使我们的代码更加灵活、可维护和可扩展。在SpringBoot项目中,策略模式可以与依赖注入机制相结合,实现更加灵活的业务逻辑处理。在本文中,...

    设计模式之策略模式 鸭子问题

    设计模式之策略模式 鸭子问题 策略模式是一种经典的设计模式,通过鸭子问题,可以让学习者更好地了解设计模式的概念和实现。策略模式的主要思想是定义一系列的算法,并将每一个算法封装起来,使它们可以相互替换。...

    桥接模式和策略模式的区别,内含可运行代码和两者详细区别

    桥接模式和策略模式是软件设计模式中的两种重要模式,它们在实现上有着相似之处,但各自的应用场景和设计理念有所不同。下面将详细阐述这两种模式的特点、区别以及它们在实际编程中的应用。 首先,桥接模式(Bridge...

    策略模式在实际项目中的应用二

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式通过定义一系列的算法,并将每一个算法封装起来,使它们可以相互替换,让算法独立于使用它的客户而变化。这种模式通常用于处理多种...

    Spring下使用策略模式

    在Spring框架中,策略模式是一种常见的设计模式,它允许我们定义一组可互换的策略,这些策略可以在运行时根据需求动态选择。这篇文章将深入探讨如何在Spring中运用策略模式,并结合源码分析其工作原理。 策略模式的...

    策略模式的实现,通过反射

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式通常用于将算法封装到不同的类中,使得可以根据需要动态选择并应用这些算法。本示例将详细介绍如何通过两种方法实现策略模式:一种...

    抽象工厂模式+工厂方法模式+策略模式+类图实现手机加工厂

    本文将探讨三个重要的设计模式:抽象工厂模式、工厂方法模式以及策略模式,并结合一个实际的场景——手机加工厂,来具体阐述它们的应用。 首先,我们来看**抽象工厂模式**。这个模式主要用于创建相关或依赖对象的...

    55-Java设计模式之策略模式与状态模式1

    Java 设计模式之策略模式与状态模式 策略模式是 Java 中的一种设计模式,它主要用于解决系统与第三方接口进行数据交互的问题。当系统需要与多种格式的数据进行交互时,使用策略模式可以很好地解决这个问题。例如,...

    策略模式封装的几个加密解密算法源码

    在"策略模式封装的几个加密解密算法源码"中,我们主要关注的是如何使用策略模式来封装常见的加密解密算法,如BASE64和MD5。 1. **BASE64编码**:BASE64是一种用于将二进制数据编码为ASCII字符的编码方式,以便在...

    策略模式的简单例子

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在软件开发中,我们经常遇到需要根据不同条件或选择执行不同算法的情况。策略模式提供了一种将算法封装到独立可互换的策略对象中,使得算法的变化独立...

    设计模式之策略模式,商场收银,封装算法

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在软件开发中,我们经常遇到需要根据不同的条件或场景来执行不同算法的情况。策略模式就是为了解决这类问题而提出的,它将每种算法封装到具有共同接口...

    js策略模式和代理模式

    策略模式和代理模式是设计模式中的两种常见模式,它们在软件开发中扮演着重要的角色,尤其是在JavaScript中,这两种模式提供了更加灵活和可维护的代码结构。 策略模式(Strategy Pattern)是一种行为设计模式,它...

    策略模式 template模式

    策略模式(Template模式) 策略模式是设计模式中的一种 객체行为型模式,它定义了一系列算法,封装每一个算法,并使它们可以互相替换。策略模式使得算法可以独立于使用它的客户而变化。 概述 在软件开发中,经常...

    Java 设计模式 策略模式

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式主要通过定义一系列的算法,并将每一个算法封装起来,使它们可以互相替换,让算法独立于使用它的客户而变化。 首先,策略模式的...

    策略模式的示例代码和思想模式

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式通常涉及接口或抽象类的实现,允许程序在运行时选择并应用不同的算法或策略。这种模式的核心在于将算法封装到独立的可互换的策略中...

    Java策略模式+案例

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式允许我们定义一组算法或策略,并将每个策略封装为一个类,使得它们可以互换,而不会影响到客户端代码。这种模式的核心在于"策略",...

    策略模式代码实现

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在策略模式中,一个类的行为或其算法可以在运行时更改。这种类型的设计模式属于行为模式。 策略模式定义了一系列的算法,并将每一个算法封装起来,使...

    设计模式——策略模式

    策略模式的设计与实现 策略模式是一种常用的设计模式,它定义了算法族,分别封装起来,让他们之间可以互相替换,此模式让算法的变化独立于使用算法的客户。策略模式的主要优点是它可以使得算法的变化独立于使用算法...

    软件设计模式策略模式实例

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在软件工程中,当一个系统需要在不同时间执行不同的算法或者行为时,策略模式就显得尤为有用。这种模式将算法封装到独立的可相互替换的策略类中,使得...

Global site tag (gtag.js) - Google Analytics