今天我们来看策略模式【Stragety Pattern【行为型】】,这个模式还是比较好理解的。策略怎么理解呢?一般是指:1. 可以实现目标的方案集合;2. 根据形势发展而制定的行动方针和斗争方法;3. 有斗争艺术,能注意方式方法。总的来说呢就是针对一个目的的不同的方法集合。这里要讲的策略模式怎么理解呢?简单的说就是对于一个类的行为或者其算法可以在运行时更改替换。
在软件系统中,一些对象使用的算法或者行为可能会经常变化,如果把这些变化的算法写到对象中的话,会使对象变得较为复杂、不易理解。那么我们如何在运行时动态的修改对象的不同的算法呢?这就使用到了策略模式。
定义一系列的算法,把它们一个个封装起来, 并且使它们可相互替换。
看上面的案例图好像和上一篇状态模式的案例图有点相似。都是包含的三个部分。具体区别我们看后面的详细介绍。
环境角色:包含持有一个Strategy抽象策略的引用。
抽象策略:定义一个公共方法,由其具体策略去实现不同算法。
具体策略:实现抽象策略接口方法。
这里我们看这么一个案例,在一个超市买东西。最后结算的时候都会问是否有会员,结算机制就有以下几种。普通用户全额计算。普通会员打95折。黄金会员打9折。钻石会员打8折。我们看下如何实现这一功能:
namespace Stragety_Pattern
{
class StragetyPattern
{
}
#region 抽象策略==================
/// <summary>
/// 抽象策略接口
/// </summary>
public interface IStragetyPattern
{
/// <summary>
/// 结算接口
/// </summary>
void Settlement(decimal Money);
}
#endregion
#region 具体策略=======================
/// <summary>
/// 无会员计算方式
/// </summary>
public class OrdinaryStragety : IStragetyPattern
{
public void Settlement(decimal Money)
{
Console.WriteLine($"不是会员,不进行折扣结算。应付款{Money}");
}
}
/// <summary>
/// 普通会员计算方式
/// </summary>
public class MemberStragety : IStragetyPattern
{
public void Settlement(decimal Money)
{
Console.WriteLine($"普通会员,打95折结算。应付款{Money*0.9M}");
}
}
/// <summary>
/// 黄金会员计算方式
/// </summary>
public class GoldMemberStragety : IStragetyPattern
{
public void Settlement(decimal Money)
{
Console.WriteLine($"黄金会员,打9折结算。应付款{Money*0.95M}");
}
}
/// <summary>
/// 钻石会员计算方式
/// </summary>
public class DiamondGoldMemberStragety : IStragetyPattern
{
public void Settlement(decimal Money)
{
Console.WriteLine($"钻石会员,打8折结算。应付款{Money*0.8M}");
}
}
#endregion
#region 环境角色
public class ContextStragety
{
private IStragetyPattern _stragety;
public ContextStragety(IStragetyPattern stragety)
{
_stragety = stragety;
}
/// <summary>
/// 调用结算方法
/// </summary>
/// <param name="Money"></param>
public void GetSettlement(decimal Money)
{
_stragety.Settlement( Money);
}
}
#endregion
}
namespace Stragety_Pattern
{
class Program
{
static void Main(string[] args)
{
decimal Account = 190.99M;
///会员计算
ContextStragety stragety = new ContextStragety(new MemberStragety());
stragety.GetSettlement(Account);
///普通结算
stragety = new ContextStragety(new OrdinaryStragety());
stragety.GetSettlement(Account);
}
}
}
这里我们针对最后结算的金额进行计算的时候是可以相互替换的。因为在具体策略中,都把算法的变化封装了起来。
1、如果在系统中有一些类,他们之间的区别就在于其行为的话。可以使用策略模式让一个对象在许多的行为中动态的选择一种行为。
2、一个系统中需要在多种算法中选择一种。
3、如果一个对象有许多的行为的话,可以简化其多重条件选择语句。避免难于维护的问题。
1、策略类之间可以自由切换,因为策略类中都是实现的抽象策略的一个方法。所以可以自由切换。
2、易于扩展,在我们新增策略的时候基本上不需要修改之前的代码。
3、对多重条件选择语句进行优化简化
1、随着策略增多,策略类会随之增加。
2、客户端必须知道所有的策略类,并且自行决定使用哪一种策略。
到这里策略模式就介绍完了。策略模式主要是针对的有一系列算法。并把他们都封装起来。他们之间可以自由切换。使这些算法的变化独立于客户端的变化。也就是把多种行为之间的变化分别进行封装起来。然后在我们调用的时候可以自由的进行切换调用。
这里我们在开篇有提到策略模式的案例图和状态模式的有点相似。这里我们重点分析一些策略模式和状态模式之间的区别吧:
1、环境角色中的任务有所不同,策略模式的环境角色中具有一种委托作用,负责根据传进来的策略调用其算法。但是状态模式中的环境角色不仅负责这行为方法的调用,还负责有记录状态变化、与具体的状态类协作。完成状态切换之后行为的切换。
2、策略模式主要解决的问题是将内部的算法的改变对外部的影响降低。保证算法的自由切换。状态模式主要解决的是状态的改变引起行为的变化、一个对象状态改变,从外界看来就好像是行为改变。
3、策略模式是一个算法的封装。这里封装的一个算法可以是有意义的对象,也可以是没有意义的逻辑片段。例如这里封装加密算法。各种加密算法,可以自由切换。算法必须是平行的。状态模式是要求一些列的状态变化随着有着行为的变化。要求拥有状态和行为。
即使受伤了,也要抬起头微笑着说,今天天气真好。