第21章 策略模式(Strategy Pattern)

原文 第21章 策略模式(Strategy Pattern)

策略模式

 

    导读:策略模式看完之后,大多数人都会感觉有点混了,包括我,感觉策略模式是一种OO思想的体现(纯属个人拙见)。

 

    概述:

 

 

    策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换。策略模式让算法独立于使用它的客户而独立变化。

    抽象策略角色: 策略类,通常由一个接口或者抽象类实现。

    具体策略角色:包装了相关的算法和行为。

    环境角色:持有一个策略类的引用,最终给客户端调用。

 

    结构图:

 

    

 

    代码举例:商品打折

    

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
     /// <summary>
     /// Strategy抽象类 抽象策略角色
     /// </summary>
     abstract  class  Strategy
     {
         public  abstract  double  Sale();
     }
     /// <summary>
     /// 具体策略角色:包装了相关的算法和行为。
     /// </summary>
     class  ConereteStrategyA : Strategy
     {
         private  double  price;
         public  ConereteStrategyA( double  price)
         {
             this .price = price;
         }
         public  override  double  Sale()
         {
             //商品打9折
             return  price * 0.9;
         }
     }
     /// <summary>
     /// 具体策略角色:包装了相关的算法和行为。
     /// </summary>
     class  ConereteStrategyB : Strategy
     {
         private  double  price;
         public  ConereteStrategyB( double  price)
         {
             this .price = price;
         }
         public  override  double  Sale()
         {
             //商品打8折
             return  price * 0.8;
         }
     }
     /// <summary>
     /// 环境角色:持有一个策略类的引用,最终给客户端调用
     /// </summary>
     class  Context
     {
         private  Strategy strategy;
         public  Context(Strategy strategy)
         {
 
             this .strategy = strategy;
         }
 
         public  double  Sale()
         {
             return  strategy.Sale();
         }
     }

  客户端调用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
     class  Program
     {
         static  void  Main( string [] args)
         {
             //打9折
             Context c1 =  new  Context( new  ConereteStrategyA(10));
             double  price = c1.Sale();
             Console.WriteLine(price);
             Console.ReadLine();
             //打八折
             Context c2 =  new  Context( new  ConereteStrategyB(10));
             double  price2 = c2.Sale();
             Console.WriteLine(price2);
             Console.ReadLine();
         }
     }

 

    优缺点:

 

    优点:

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

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

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

 

    缺点:

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

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

 

    适用场景:

 

 

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

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

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

    

    设计模式系列文章入口:http://www.diyibk.com/post/39.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值