概述

  如果在开发过程中,出现大量的if else或者switch case 语句,如果这些语句块中的代码并不是包含业务逻辑,只是单纯的分流方法,那么,每一个语句块中都是一个算法或者叫策略。

背景

  比如在最近项目中遇到的问题。一个二维码字符串解析的方法:

    微信的二维码扫描结果包含“WeChat”,解析规则是拿着文本到微信服务器解析,返回解析对象。

    支付宝二维码扫描结果包含“Alipay”,解析规则是使用“->”分割字符串得到解析对象。

  最简单快速的代码就是直接if else判断:

  

         /// <summary>
/// 解析方法
/// </summary>
/// <param name="text">扫描得到的文本</param>
public void AnalysisAction(string text)
{
//微信解析方法
if (text.Contains("WeChat"))
{
//拿着text到微信服务器解析,返回解析对象。
}
//支付宝解析方法
else if (text.Contains("Alipay"))
{
//使用->分割,得到解析对象。
}
}

问题

  当然使用这种方式是可以的,但是如果以后又加入一种扫码解析方法:

  中国联通二维码扫描文本中包含“Unicom”,解析规则为以“:”分割,得到解析对象。

  那么你就要继续添加else if(text.Contains("Unicom"))。每次增加一个新的扫描解析规则,你都要去增加else if判断,这种是面向过程的体验,属于硬编码。这也违反了面向对象的开闭原则。

改进(抽象)

  我们可以使用策略模式来改进代码。义一系列的算法,把每一个算法封装起来, 并且使它们可相互替换。本模式使得算法可独立于使用它的客户而变化。

  抽象出一个扫描解析接口,定义一个解析方法。然后分别定义微信和支付宝的解析方法集成接口。

  策略模式的uml图如下:

  

 
    /// <summary>
/// 扫描解析规则抽象接口
/// </summary>
public interface IStrategy
{
/// <summary>
/// 扫描解析方法策略
/// </summary>
/// <param name="text"></param>
void AnalysisAction(string text);
}

扫描解析规则抽象接口

    /// <summary>
/// 微信扫描解析规则
/// </summary>
public class StrategyWeChat : IStrategy
{
/// <summary>
/// 扫描策略
/// </summary>
/// <param name="text"></param>
public void AnalysisAction(string text)
{
//拿着text到微信服务器解析,返回解析对象。
}
}

微信扫描策略

    /// <summary>
/// 支付宝扫描解析规则
/// </summary>
public class StrategyAlipay : IStrategy
{
/// <summary>
/// 扫描策略
/// </summary>
/// <param name="text"></param>
public void AnalysisAction(string text)
{
//使用->分割,得到解析对象。
}
}

支付宝扫描解析策略

    /// <summary>
/// 接口管理类
/// </summary>
public class StrategyContext
{
private IStrategy strategy;
/// <summary>
/// 外层调用的时候决定使用哪个扫描策略
/// </summary>
/// <param name="strategy"></param>
public StrategyContext(IStrategy strategy)
{
this.strategy = strategy;
}
public void AnalysisAction(string text)
{
strategy.AnalysisAction(text);
}
}

管理类

这样我们的业务逻辑就可以这样写:

        private StrategyContext Context;
/// <summary>
/// 解析方法
/// </summary>
/// <param name="text">扫描得到的文本</param>
public void AnalysisAction(string text)
{ //微信解析方法
if (text.Contains("WeChat"))
{
Context = new StrategyContext(new StrategyWeChat());
}
//支付宝解析方法
else if (text.Contains("Alipay"))
{
Context = new StrategyContext(new StrategyAlipay());
}
Context.AnalysisAction(text);
}

  我们将具体的解析规则放到了具体的实现类中,但是我们并没有消灭If else,如果在添加联通的扫码解析的话,还是需要修改代码,添加else。

  这就是策略模式的缺点,必须知道要使用的具体的策略,也就是有的人说的还是要使用if else。

升级改造

  因为前面说到了策略模式的缺点。如果就是要消灭if else呢?我们可以将决定使用策略的决定权放到具体策略实现类中。

     /// <summary>
/// 扫描解析规则抽象接口
/// </summary>
public interface IStrategy
{
/// <summary>
/// 是否可以解析
/// </summary>
bool Analysisable { get; } /// <summary>
/// 扫描解析方法策略
/// </summary>
/// <param name="text"></param>
void AnalysisAction();
}

扫描解析规则抽象接口添加是否可以解析属性

    /// <summary>
/// 微信扫描解析规则
/// </summary>
public class StrategyWeChat : IStrategy
{
private string _text;
public StrategyWeChat(string text)
{
this._text = text;
}
public bool Analysisable
{
get { return _text.Contains("WeChat"); }
} /// <summary>
/// 扫描策略
/// </summary>
/// <param name="text"></param>
public void AnalysisAction()
{
//拿着_text到微信服务器解析,返回解析对象。
}
}

微信

    /// <summary>
/// 支付宝扫描解析规则
/// </summary>
public class StrategyAlipay : IStrategy
{
private string _text;
public StrategyAlipay(string text)
{
this._text = text;
}
public bool Analysisable
{
get { return _text.Contains("Alipay"); }
}
/// <summary>
/// 扫描策略
/// </summary>
/// <param name="text"></param>
public void AnalysisAction()
{
//使用->分割,得到解析对象。
}
}

支付宝

     public class StrategyContext2
{
private readonly IList<IStrategy> strategyList = new List<IStrategy>();
/// <summary>
/// 将所有策略都方法
/// </summary>
/// <param name="text"></param>
public StrategyContext2(string text)
{
strategyList.Add(new StrategyWeChat(text));
strategyList.Add(new StrategyAlipay(text));
}
/// <summary>
/// 调用具体的策略类实现扫码解析方法
/// </summary>
public void AnalysisAction()
{
foreach (var item in strategyList)
{
if (item.Analysisable)//判断当前策略类是否可以处理
item.AnalysisAction();
}
}
}

Context

         private StrategyContext2 Context;
public void AnalysisAction(string text)
{
Context = new StrategyContext2(text);
Context.AnalysisAction();//自动实现解析,不用关心使用哪种策略
}

这样我们就想决定权放到了具体策略类本身中。消灭了If else。如果再添加联通扫码策略的时候,只需要添加联通的具体扫描策略,然后在context构造函数中把他加入到策略集合中。

但是这样我们还是修改了context代码。如果继续想不修改context代码呢?

继续升级

我们可以使用反射,将所有策略实现类都反射出来,添加到策略集合中。那么我们的context类可以这样写:

    public class StrategyContext3
{
private readonly IList<IStrategy> strategyList = new List<IStrategy>();
/// <summary>
/// 将所有策略都方法
/// </summary>
/// <param name="text"></param>
public StrategyContext3(string text)
{
//查询程序集
Assembly assembly = Assembly.GetExecutingAssembly();
//找出继承扫描策略接口的类
IEnumerable<Type> types = assembly.GetTypes().Where(c => c.GetInterface("IStrategy") != null);
foreach (var t in types)
{
object[] parameters = new object[];
parameters[] = text;
//创建类的实例
strategyList.Add((IStrategy)Activator.CreateInstance(t, parameters));
}
}
/// <summary>
/// 调用具体的策略类实现扫码解析方法
/// </summary>
public void AnalysisAction()
{
foreach (var item in strategyList)
{
if (item.Analysisable)//判断当前策略类是否可以处理
item.AnalysisAction();
}
}
}

context

应用场景

1、 多个类只区别在表现行为不同,可以使用Strategy模式,在运行时动态选择具体要执行的行为。
2、 需要在不同情况下使用不同的策略(算法),或者策略还可能在未来用其它方式来实现。
3、 对客户隐藏具体策略(算法)的实现细节,彼此完全独立。

总结

  优点:策略模式是对算法的封装,它把算法的责任和算法本身分割开,委派给不同的对象管理。

  缺点:客户端必须知道所有的策略类,并自行决定使用哪一个策略类。这就意味着客户端必须理解这些算法的区别,以便适时选择恰当的算法类。换言之,策略模式只适用于客户端知道所有的算法或行为的情况。 这也就是我们所说的if else并没有真正的被消灭。

  改进:我们可以将决定使用哪种策略的权利放到策略类本身中,让策略自己决定到底是不是用我自己的方法。从而实现消灭if else。

     可以通过反射,反射出所有的策略。这样比较符合开闭原则。

设计模式-策略模式Strategy以及消灭if else的更多相关文章

  1. [.net 面向对象程序设计深入](26)实战设计模式——策略模式 Strategy (行为型)

    [.net 面向对象程序设计深入](26)实战设计模式——策略模式 Strategy (行为型) 1,策略模式定义 策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换.策略模 ...

  2. 设计模式--策略模式(strategy)

    1.策略模式(strategy ['strætədʒi]) 我的理解是:方案候选模式 (反正关键就是有很多的候选,哈哈) 看了很多例子,都是在说鸭子的,那个例子很好,在这里可以看 他们生产鸭子,我们就 ...

  3. 设计模式 - 策略模式(Strategy Pattern) 具体解释

    策略模式(Strategy Pattern) 具体解释 本文地址: http://blog.csdn.net/caroline_wendy/article/details/26577879 本文版权全 ...

  4. java设计模式 策略模式Strategy

    本章讲述java设计模式中,策略模式相关的知识点. 1.策略模式定义 策略模式,又叫算法簇模式,就是定义了不同的算法族,并且之间可以互相替换,此模式让算法的变化独立于使用算法的客户.策略模式属于对象的 ...

  5. 设计模式——策略模式(Strategy Pattern)

    写在前面: 直接将书中的例子用来作为记录自己学习的成果,不知道这样好不好,如果给原作者带来什么不利的影响不妨告知一声,我及时删掉. UML图: 抽象策略:Strategy package com.cn ...

  6. 说说设计模式~策略模式(Strategy)

    返回目录 策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换.而对于客户端(UI)来说,可以通过IOC再配合工厂模块,实现动态策略的切换,策略模块通常于一个抽象策略对象(in ...

  7. C#设计模式——策略模式(Strategy Pattern)

    一.概述我们来实现一个企业的工资系统,该企业中不同级别的员工工资算法都不相同,针对该问题,最容易想到的莫过于在代码中堆积一大堆if…else…语句或者是switch…case…语句.如果该企业中不同级 ...

  8. 设计模式---策略模式Strategy(对象行为型)

    1. 概述 策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换.策略模式让算法独立于使用它的客户而独立变化. 策略模式是对算法的封装,它把算法的责任和算法本身分割开,委派给不 ...

  9. 大话设计模式--策略模式 strategy -- C++实现实例

    1. 策略模式: 它定义了算法家族, 分别封装起来,使他们之间可以相互替换,此模式让算法变化, 不会影响到使用算法的客户. 用相同的方法调用不同的算法,减少各种算法类与使用算法类之间的耦合. 实例中策 ...

随机推荐

  1. iframe访问子页面方法

    在Iframe中调用子页面的Js函数 调用IFRAME子页面的JS函数 说明:假设有2个页面,index.html和inner.html.其中index.html中有一个iframe,这个iframe ...

  2. 【踩坑记录】记一次MySQL主从复制延迟的坑

    最近开发中遇到的一个MySQL主从延迟的坑,记录并总结,避免再次犯同样的错误. 情景 一个活动信息需要审批,审批之后才能生效.因为之后活动要编辑,编辑后也可能触发审批,审批中展示的是编辑前的活动内容, ...

  3. HTML特殊布局--------双飞翼布局

    今天看到以前写的一篇布局的例子分享给大家,双飞翼布局. 什么是双飞翼布局?? 1.三列布局,中间宽度自适应,两边固定宽度; 2.中间栏在浏览器中优先展示渲染: 双飞翼布局的原理: 中间的盒子定100% ...

  4. 找到你在网页中缓存起来的flash文件

    通过IE浏览器工具->Internet选项->常规->设置->Internet临时文件->查看文件(找到你在网页中缓存起来的flash文件)

  5. HBuilder使用方法

    /*注:本教程针对HBuilder5.0.0,制作日期2014-12-31*/创建HTML结构: h 8 (敲h激活代码块列表,按8选择第8个项目,即HTML代码块,或者敲h t Enter)中途换行 ...

  6. Jquery-鼠标事件

    鼠标事件是在用户移动鼠标光标或者使用任意鼠标键点击时触发的.(1):click事件:click事件于用户在元素敲击鼠标左键,并在相同元素上松开左键时触发.        $('p').click(fu ...

  7. Linux之用户管理--初级上

    管理用户命令汇总 命令 注释说明(特殊颜色的必须掌握) useradd增 同adduser命令,执行此命令可在系统中添加用户.(更改4个用户文件) userdel删 执行此命令可删除用户及相关用户的配 ...

  8. ReadAndWriteBinaryFile

    package JBJADV003;import java.io.FileInputStream;import java.io.DataInputStream;import java.io.EOFEx ...

  9. Java IO设计模式(装饰模式与适配器模式)

    01. 装饰模式 1. 定义 Decorator装饰器,就是动态地给一个对象添加一些额外的职责,动态扩展,和下面继承(静态扩展)的比较.因此,装饰器模式具有如下的特征: 它必须持有一个被装饰的对象(作 ...

  10. 拥抱.NET Core系列:Logging (1)

    在之前我们简单介绍了 .NET Core 中的 DI组件,没来及了解的童鞋可以翻翻我之前的文章. 接下来会对 .NET Core 中的 Logging 进行介绍. 本文中使用了"Micros ...