设计模式—策略(Strategy)模式

简介: 设计模式—策略(Strategy)模式

 一、概述

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

使用策略模式可以把行为和环境分割开来。环境类负责维持和查询行为类,各种算法则在具体策略类(ConcreteStrategy)中提供。由于算法和环境独立开来,算法的增减、修改都不会影响环境和客户端。当出现新的促销折扣或现有的折扣政策出现变化时,只需要实现新的策略类,并在客户端登记即可。策略模式相当于"可插入式(Pluggable)的算法"。

二、策略模式的结构

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

策略又称做政策(Policy)模式【GOF95】。下面是一个示意性的策略模式结构图:

image.gif

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

    • 环境(Context)角色:持有一个Strategy类的引用。
      • 抽象策略(Strategy)角色:这是一个抽象角色,通常由一个接口或抽象类实现。此角色给出所有的具体策略类所需的接口。
        • 具体策略(ConcreteStrategy)角色:包装了相关的算法或行为。

        三、 示意性源代码

        public abstract class Strategy {
            abstract public void AlgorithmInterface();
        }
        public class ConcreteStrategyA extends Strategy {
            // Methods
            @Override
            public void AlgorithmInterface() {
                System.out.print("Called ConcreteStrategyA.AlgorithmInterface()");
            }
        }
        // "ConcreteStrategyB"
        public class ConcreteStrategyB extends Strategy {
            // Methods
            public void AlgorithmInterface() {
                System.out.print("Called ConcreteStrategyB.AlgorithmInterface()");
            }
        }
        // "ConcreteStrategyC"
        public class ConcreteStrategyC extends Strategy {
            // Methods
            @Override
            public void AlgorithmInterface() {
                System.out.print("Called ConcreteStrategyC.AlgorithmInterface()");
            }
        }
        // "Context"
        public class Context {
            // Fields
            Strategy strategy;
            // Constructors
            public Context(Strategy strategy) {
                this.strategy = strategy;
            }
            // Methods
            public void ContextInterface() {
                strategy.AlgorithmInterface();
            }
        }
        /// <summary>
        /// Client test
        /// </summary>
        public class Client {
            public static void Main(String[] args) {
                // Three contexts following different strategies
                Context c = new Context(new ConcreteStrategyA());
                c.ContextInterface();
                Context d = new Context(new ConcreteStrategyB());
                d.ContextInterface();
                Context e = new Context(new ConcreteStrategyC());
                e.ContextInterface();
            }
        }

        image.gif

        四、 何时使用何种具体策略角色

        在学习策略模式时,学员常问的一个问题是:为什么不能从策略模式中看出哪一个具体策略适用于哪一种情况呢?

        答案非常简单,策略模式并不负责做这个决定。换言之,应当由客户端自己决定在什么情况下使用什么具体策略角色。策略模式仅仅封装算法,提供新算法插入到已有系统中,以及老算法从系统中"退休"的方便,策略模式并不决定在何时使用何种算法。

        五、 一个实际应用策略模式的例子

        下面的例子利用策略模式在排序对象中封装了不同的排序算法,这样以便允许客户端动态的替换排序策略(包括Quicksort、Shellsort和Mergesort)。

        abstract class SortStrategy {
            // Methods
            public abstract void Sort(List<String> list);
        }
        // "ConcreteStrategy"
        class QuickSort extends SortStrategy {
            // Methods
            @Override
            public void Sort(List<String> list) {
                // Default is Quicksort
                System.out.print("QuickSorted list ");
            }
        }
        // "ConcreteStrategy"
        class ShellSort extends SortStrategy {
            // Methods
            @Override
            public void Sort(List<String> list) {
                //list.ShellSort();
                System.out.print("ShellSorted list ");
            }
        }
        // "ConcreteStrategy"
        class MergeSort extends SortStrategy {
            // Methods
            @Override
            public void Sort(List<String> list) {
                //list.MergeSort();
                System.out.print("MergeSorted list ");
            }
        }
        // "Context"
        class SortedList {
            // Fields
            private List<String> list = new ArrayList<>();
            private SortStrategy sortstrategy;
            // Constructors
            public void SetSortStrategy(SortStrategy sortstrategy) {
                this.sortstrategy = sortstrategy;
            }
            // Methods
            public void Sort() {
                sortstrategy.Sort(list);
            }
            public void Add(String name) {
                list.add(name);
            }
            public void Display() {
                System.out.print(" name");
            }
        }
        /// <summary>
        /// StrategyApp test
        /// </summary>
        public class StrategyApp {
            public static void Main(String[] args) {
                // Two contexts following different strategies
                SortedList studentRecords = new SortedList();
                studentRecords.Add("Samual");
                studentRecords.Add("Jimmy");
                studentRecords.Add("Sandra");
                studentRecords.Add("Anna");
                studentRecords.Add("Vivek");
                studentRecords.SetSortStrategy(new QuickSort());
                studentRecords.Sort();
                studentRecords.Display();
            }
        }

        image.gif

        六、 在什么情况下应当使用策略模式

        在下面的情况下应当考虑使用策略模式:

        1. 如果在一个系统里面有许多类,它们之间的区别仅在于它们的行为,那么使用策略模式可以动态地让一个对象在许多行为中选择一种行为。

        2. 一个系统需要动态地在几种算法中选择一种。那么这些算法可以包装到一个个的具体算法类里面,而这些具体算法类都是一个抽象算法类的子类。换言之,这些具体算法类均有统一的接口,由于多态性原则,客户端可以选择使用任何一个具体算法类,并只持有一个数据类型是抽象算法类的对象。

        3. 一个系统的算法使用的数据不可以让客户端知道。策略模式可以避免让客户端涉及到不必要接触到的复杂的和只与算法有关的数据。

        4. 如果一个对象有很多的行为,如果不用恰当的模式,这些行为就只好使用多重的条件选择语句来实现。此时,使用策略模式,把这些行为转移到相应的具体策略类里面,就可以避免使用难以维护的多重条件选择语句,并体现面向对象设计的概念。

        七、 策略模式的优点和缺点

        策略模式有很多优点和缺点。它的优点有:

        1. 策略模式提供了管理相关的算法族的办法。策略类的等级结构定义了一个算法或行为族。恰当使用继承可以把公共的代码移到父类里面,从而避免重复的代码。

        2. 策略模式提供了可以替换继承关系的办法。继承可以处理多种算法或行为。如果不是用策略模式,那么使用算法或行为的环境类就可能会有一些子类,每一个子类提供一个不同的算法或行为。但是,这样一来算法或行为的使用者就和算法或行为本身混在一起。决定使用哪一种算法或采取哪一种行为的逻辑就和算法或行为的逻辑混合在一起,从而不可能再独立演化。继承使得动态改变算法或行为变得不可能。

        3. 使用策略模式可以避免使用多重条件转移语句。多重转移语句不易维护,它把采取哪一种算法或采取哪一种行为的逻辑与算法或行为的逻辑混合在一起,统统列在一个多重转移语句里面,比使用继承的办法还要原始和落后。

        策略模式的缺点有:

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

        2. 策略模式造成很多的策略类。有时候可以通过把依赖于环境的状态保存到客户端里面,而将策略类设计成可共享的,这样策略类实例可以被不同客户端使用。换言之,可以使用享元模式来减少对象的数量。

        目录
        相关文章
        |
        1月前
        |
        设计模式 安全 Java
        Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
        Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
        |
        3月前
        |
        设计模式 数据库连接 PHP
        PHP中的设计模式:提升代码的可维护性与扩展性在软件开发过程中,设计模式是开发者们经常用到的工具之一。它们提供了经过验证的解决方案,可以帮助我们解决常见的软件设计问题。本文将介绍PHP中常用的设计模式,以及如何利用这些模式来提高代码的可维护性和扩展性。我们将从基础的设计模式入手,逐步深入到更复杂的应用场景。通过实际案例分析,读者可以更好地理解如何在PHP开发中应用这些设计模式,从而写出更加高效、灵活和易于维护的代码。
        本文探讨了PHP中常用的设计模式及其在实际项目中的应用。内容涵盖设计模式的基本概念、分类和具体使用场景,重点介绍了单例模式、工厂模式和观察者模式等常见模式。通过具体的代码示例,展示了如何在PHP项目中有效利用设计模式来提升代码的可维护性和扩展性。文章还讨论了设计模式的选择原则和注意事项,帮助开发者在不同情境下做出最佳决策。
        |
        1月前
        |
        设计模式 开发者 Python
        Python编程中的设计模式:工厂方法模式###
        本文深入浅出地探讨了Python编程中的一种重要设计模式——工厂方法模式。通过具体案例和代码示例,我们将了解工厂方法模式的定义、应用场景、实现步骤以及其优势与潜在缺点。无论你是Python新手还是有经验的开发者,都能从本文中获得关于如何在实际项目中有效应用工厂方法模式的启发。 ###
        |
        1月前
        |
        设计模式 安全 Java
        Kotlin - 改良设计模式 - 构建者模式
        Kotlin - 改良设计模式 - 构建者模式
        |
        1月前
        |
        设计模式 安全 Java
        Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
        Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
        45 1
        |
        2月前
        |
        设计模式 Java Kotlin
        Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
        本教程详细讲解Kotlin语法,适合希望深入了解Kotlin的开发者。对于快速学习Kotlin语法,推荐查看“简洁”系列教程。本文重点介绍了构建者模式在Kotlin中的应用与改良,包括如何使用具名可选参数简化复杂对象的创建过程,以及如何在初始化代码块中对参数进行约束和校验。
        32 3
        |
        2月前
        |
        设计模式 监控 算法
        Java设计模式梳理:行为型模式(策略,观察者等)
        本文详细介绍了Java设计模式中的行为型模式,包括策略模式、观察者模式、责任链模式、模板方法模式和状态模式。通过具体示例代码,深入浅出地讲解了每种模式的应用场景与实现方式。例如,策略模式通过定义一系列算法让客户端在运行时选择所需算法;观察者模式则让多个观察者对象同时监听某一个主题对象,实现松耦合的消息传递机制。此外,还探讨了这些模式与实际开发中的联系,帮助读者更好地理解和应用设计模式,提升代码质量。
        Java设计模式梳理:行为型模式(策略,观察者等)
        |
        3月前
        |
        设计模式 数据库连接 PHP
        PHP中的设计模式:如何提高代码的可维护性与扩展性在软件开发领域,PHP 是一种广泛使用的服务器端脚本语言。随着项目规模的扩大和复杂性的增加,保持代码的可维护性和可扩展性变得越来越重要。本文将探讨 PHP 中的设计模式,并通过实例展示如何应用这些模式来提高代码质量。
        设计模式是经过验证的解决软件设计问题的方法。它们不是具体的代码,而是一种编码和设计经验的总结。在PHP开发中,合理地使用设计模式可以显著提高代码的可维护性、复用性和扩展性。本文将介绍几种常见的设计模式,包括单例模式、工厂模式和观察者模式,并通过具体的例子展示如何在PHP项目中应用这些模式。
        |
        3月前
        |
        设计模式 Java Spring
        spring源码设计模式分析-代理设计模式(二)
        spring源码设计模式分析-代理设计模式(二)
        |
        2月前
        |
        设计模式 安全 Java
        Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
        Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
        43 0