一、 策略(Strategy)模式
策略模式的用意是针对一组算法,将每一个算法封装到具有共同接口的独立的类中,从而使得它们可以相互替换。策略模式使得算法可以在不影响到客户端的情况下发生变化。
假设现在要设计一个贩卖各类书籍的电子商务网站的购物车(Shopping Cat)系统。一个最简单的情况就是把所有货品的单价乘上数量,但是实际情况肯定比这要复杂。比如,本网站可能对所有的教材类图书实行每本一元的折扣;对连环画类图书提供每本7%的促销折扣,而对非教材类的计算机图书有3%的折扣;对其余的图书没有折扣。由于有这样复杂的折扣算法,使得价格计算问题需要系统地解决。
使用策略模式可以把行为和环境分割开来。环境类负责维持和查询行为类,各种算法则在具体策略类(ConcreteStrategy)中提供。由于算法和环境独立开来,算法的增减、修改都不会影响环境和客户端。当出现新的促销折扣或现有的折扣政策出现变化时,只需要实现新的策略类,并在客户端登记即可。策略模式相当于"可插入式(Pluggable)的算法"。
二、 策略模式的结构
策略模式是对算法的包装,是把使用算法的责任和算法本身分割开,委派给不同的对象管理。策略模式通常把一个系列的算法包装到一系列的策略类里面,作为一个抽象策略类的子类。用一句话来说,就是:"准备一组算法,并将每一个算法封装起来,使得它们可以互换。"
策略又称做政策(Policy)模式【GOF95】。下面是一个示意性的策略模式结构图:
这个模式涉及到三个角色:
- 环境(Context)角色:持有一个Strategy类的引用。
- 抽象策略(Strategy)角色:这是一个抽象角色,通常由一个接口或抽象类实现。此角色给出所有的具体策略类所需的接口。
- 具体策略(ConcreteStrategy)角色:包装了相关的算法或行为。
三、 示意性源代码
1 // Strategy pattern -- Structural example 2 using System; 3 4 // "Strategy" 5 abstract class Strategy 6 { 7 // Methods 8 abstract public void AlgorithmInterface(); 9 } 10 11 // "ConcreteStrategyA" 12 class ConcreteStrategyA : Strategy 13 { 14 // Methods 15 override public void AlgorithmInterface() 16 { 17 Console.WriteLine("Called ConcreteStrategyA.AlgorithmInterface()"); 18 } 19 } 20 21 // "ConcreteStrategyB" 22 class ConcreteStrategyB : Strategy 23 { 24 // Methods 25 override public void AlgorithmInterface() 26 { 27 Console.WriteLine("Called ConcreteStrategyB.AlgorithmInterface()"); 28 } 29 } 30 31 // "ConcreteStrategyC" 32 class ConcreteStrategyC : Strategy 33 { 34 // Methods 35 override public void AlgorithmInterface() 36 { 37 Console.WriteLine("Called ConcreteStrategyC.AlgorithmInterface()"); 38 } 39 } 40 41 // "Context" 42 class Context 43 { 44 // Fields 45 Strategy strategy; 46 47 // Constructors 48 public Context( Strategy strategy ) 49 { 50 this.strategy = strategy; 51 } 52 53 // Methods 54 public void ContextInterface() 55 { 56 strategy.AlgorithmInterface(); 57 } 58 } 59 60 /// <summary> 61 /// Client test 62 /// </summary> 63 public class Client 64 { 65 public static void Main( string[] args ) 66 { 67 // Three contexts following different strategies 68 Context c = new Context( new ConcreteStrategyA() ); 69 c.ContextInterface(); 70 71 Context d = new Context( new ConcreteStrategyB() ); 72 d.ContextInterface(); 73 74 Context e = new Context( new ConcreteStrategyC() ); 75 e.ContextInterface(); 76 } 77 }
四、 何时使用何种具体策略角色
在学习策略模式时,学员常问的一个问题是:为什么不能从策略模式中看出哪一个具体策略适用于哪一种情况呢?
答案非常简单,策略模式并不负责做这个决定。换言之,应当由客户端自己决定在什么情况下使用什么具体策略角色。策略模式仅仅封装算法,提供新算法插入到已有系统中,以及老算法从系统中"退休"的方便,策略模式并不决定在何时使用何种算法。
五、 一个实际应用策略模式的例子
下面的例子利用策略模式在排序对象中封装了不同的排序算法,这样以便允许客户端动态的替换排序策略(包括Quicksort、Shellsort和Mergesort)。
1 // Strategy pattern -- Real World example 2 using System; 3 using System.Collections; 4 5 // "Strategy" 6 abstract class SortStrategy 7 { 8 // Methods 9 abstract public void Sort( ArrayList list ); 10 } 11 12 // "ConcreteStrategy" 13 class QuickSort : SortStrategy 14 { 15 // Methods 16 public override void Sort(ArrayList list ) 17 { 18 list.Sort(); // Default is Quicksort 19 Console.WriteLine("QuickSorted list "); 20 } 21 } 22 23 // "ConcreteStrategy" 24 class ShellSort : SortStrategy 25 { 26 // Methods 27 public override void Sort(ArrayList list ) 28 { 29 //list.ShellSort(); 30 Console.WriteLine("ShellSorted list "); 31 } 32 } 33 34 // "ConcreteStrategy" 35 class MergeSort : SortStrategy 36 { 37 // Methods 38 public override void Sort( ArrayList list ) 39 { 40 //list.MergeSort(); 41 Console.WriteLine("MergeSorted list "); 42 } 43 } 44 45 // "Context" 46 class SortedList 47 { 48 // Fields 49 private ArrayList list = new ArrayList(); 50 private SortStrategy sortstrategy; 51 52 // Constructors 53 public void SetSortStrategy( SortStrategy sortstrategy ) 54 { 55 this.sortstrategy = sortstrategy; 56 } 57 58 // Methods 59 public void Sort() 60 { 61 sortstrategy.Sort( list ); 62 } 63 64 public void Add( string name ) 65 { 66 list.Add( name ); 67 } 68 69 public void Display() 70 { 71 foreach( string name in list ) 72 Console.WriteLine( " " + name ); 73 } 74 } 75 76 /// <summary> 77 /// StrategyApp test 78 /// </summary> 79 public class StrategyApp 80 { 81 public static void Main( string[] args ) 82 { 83 // Two contexts following different strategies 84 SortedList studentRecords = new SortedList( ); 85 studentRecords.Add( "Samual" ); 86 studentRecords.Add( "Jimmy" ); 87 studentRecords.Add( "Sandra" ); 88 studentRecords.Add( "Anna" ); 89 studentRecords.Add( "Vivek" ); 90 91 studentRecords.SetSortStrategy( new QuickSort() ); 92 studentRecords.Sort(); 93 studentRecords.Display(); 94 } 95 }
六、 在什么情况下应当使用策略模式
在下面的情况下应当考虑使用策略模式:
1. 如果在一个系统里面有许多类,它们之间的区别仅在于它们的行为,那么使用策略模式可以动态地让一个对象在许多行为中选择一种行为。
2. 一个系统需要动态地在几种算法中选择一种。那么这些算法可以包装到一个个的具体算法类里面,而这些具体算法类都是一个抽象算法类的子类。换言之,这些具体算法类均有统一的接口,由于多态性原则,客户端可以选择使用任何一个具体算法类,并只持有一个数据类型是抽象算法类的对象。
3. 一个系统的算法使用的数据不可以让客户端知道。策略模式可以避免让客户端涉及到不必要接触到的复杂的和只与算法有关的数据。
4. 如果一个对象有很多的行为,如果不用恰当的模式,这些行为就只好使用多重的条件选择语句来实现。此时,使用策略模式,把这些行为转移到相应的具体策略类里面,就可以避免使用难以维护的多重条件选择语句,并体现面向对象设计的概念。
七、 策略模式的优点和缺点
策略模式有很多优点和缺点。它的优点有:
1. 策略模式提供了管理相关的算法族的办法。策略类的等级结构定义了一个算法或行为族。恰当使用继承可以把公共的代码移到父类里面,从而避免重复的代码。
2. 策略模式提供了可以替换继承关系的办法。继承可以处理多种算法或行为。如果不是用策略模式,那么使用算法或行为的环境类就可能会有一些子类,每一个子类提供一个不同的算法或行为。但是,这样一来算法或行为的使用者就和算法或行为本身混在一起。决定使用哪一种算法或采取哪一种行为的逻辑就和算法或行为的逻辑混合在一起,从而不可能再独立演化。继承使得动态改变算法或行为变得不可能。
3. 使用策略模式可以避免使用多重条件转移语句。多重转移语句不易维护,它把采取哪一种算法或采取哪一种行为的逻辑与算法或行为的逻辑混合在一起,统统列在一个多重转移语句里面,比使用继承的办法还要原始和落后。
策略模式的缺点有:
1. 客户端必须知道所有的策略类,并自行决定使用哪一个策略类。这就意味着客户端必须理解这些算法的区别,以便适时选择恰当的算法类。换言之,策略模式只适用于客户端知道所有的算法或行为的情况。
2. 策略模式造成很多的策略类。有时候可以通过把依赖于环境的状态保存到客户端里面,而将策略类设计成可共享的,这样策略类实例可以被不同客户端使用。换言之,可以使用享元模式来减少对象的数量。
八、 其它
策略模式与很多其它的模式都有着广泛的联系。Strategy很容易和Bridge模式相混淆。虽然它们结构很相似,但它们却是为解决不同的问题而设计的。Strategy模式注重于算法的封装,而Bridge模式注重于分离抽象和实现,为一个抽象体系提供不同的实现。Bridge模式与Strategy模式都很好的体现了"Favor composite over inheritance"的观点。
推荐大家读一读《IoC 容器和Dependency Injection 模式》,作者Martin Fowler。网上可以找到中文版的PDF文件。为策略模式的实施提供了一个非常好的方案。