设计模式之——策略模式(Strategy Pattern)及在Android中的应用

相信大家都用过计算器,输入一个数,然后输入运算符,然后再输入一个数,就会根据不同的运算符做不同的运算。

最直接的加减法:

public class Calculator {
    //加符号
    private final static String ADD_SYMBOL = "+";
    //减符号
    private final static String SUB_SYMBOL = "-";
    public int exec(int a,int b,String symbol){
        int result =0;
        if(symbol.equals(ADD_SYMBOL)){
            result = this.add(a, b);
        }else if(symbol.equals(SUB_SYMBOL)){
            result = this.sub(a, b);
        }
        return result;
    }
    //加法运算
    private int add(int a,int b){
        return a+b;
    }
    //减法运算
    private int sub(int a,int b){
        return a-b;
    }
}

用户使用:

public class Client {
    public static void main(String[] args) {
        //输入的两个参数是数字
        int a = Integer.parseInt(args[0]);
        String symbol = args[1]; //符号
        int b = Integer.parseInt(args[2]);
        System.out.println("输入的参数为:"+Arrays.toString(args));
        //生成一个运算器
        Calculator cal = new Calculator();
        System.out.println("运行结果为:"+a + symbol + b + "=" + cal.exec(a, b, symbol));
    }
}

这是最简单直接的代码,有什么问题吗?假如用户需要这个计算器支持乘法呢?就要改Calculator类,明显违背了开闭原则,系统也不利于维护。

那么怎么设计成可以扩展的代码呢?就需要策略模式了。

策略模式类图

定义:策略模式也叫政策模式,定义一组算法,将每个算法都封装起来,并且使它们之间可以互换。

这个定义是非常明确、清晰的,“定义一组算法”,看看加减法和乘法是不是三个算
法?“将每个算法都封装起来”,那么我们定义一个类,封装算法,可以互换,是不是多态的特征呢?我们用代码把这个定义实现下:

//抽象策略
interface Calculator {
    public int exec(int a,int b);
}
// 具体策略

public class Add implements Calculator {
    // 加法运算
    public int exec(int a, int b) {
        return a+b;
    }
}
public class Sub implements Calculator {
    //减法运算
    public int exec(int a, int b) {
        return a-b;
    }
}

策略定义好了,然后定义一个Context封装类,其作用是承装三个策略,根据不同的需要替换:

public class Context {
    private Calculator cal = null;
    public Context(Calculator _cal){
        this.cal = _cal;
    }
    public int exec(int a,int b,String symbol){
        return this.cal.exec(a, b);
    }
}

用户使用:

public class Client {
    //加符号
    public final static String ADD_SYMBOL = "+";
    //减符号
    public final static String SUB_SYMBOL = "-";
    public static void main(String[] args) {
        //输入的两个参数是数字
        int a = Integer.parseInt(args[0]);
        String symbol = args[1]; //符号
        int b = Integer.parseInt(args[2]);
        System.out.println("输入的参数为:"+Arrays.toString(args));
        //上下文
        Context context = null;
        //判断初始化哪一个策略
        if(symbol.equals(ADD_SYMBOL)){
            context = new Context(new Add());
        }else if(symbol.equals(SUB_SYMBOL)){
            context = new Context(new Sub());
        }
        System.out.println("运行结果为:"+a+symbol+b+"="+context.exec(a,b,symbol));
    }
}

需要增加乘法呢?实现Calculator ,增加乘法算法,直接替换就ok了:

public class Mul implements Calculator {
    //乘法运算
    public int exec(int a, int b) {
        return a*b;
    }
}

public class Client {
    //加符号
    public final static String ADD_SYMBOL = "+";
    //减符号
    public final static String SUB_SYMBOL = "-";
    //乘符号
    public final static String MUL_SYMBOL = "*";
    public static void main(String[] args) {
        //输入的两个参数是数字
        int a = Integer.parseInt(args[0]);
        String symbol = args[1]; //符号
        int b = Integer.parseInt(args[2]);
        System.out.println("输入的参数为:"+Arrays.toString(args));
        //上下文
        Context context = null;
        //判断初始化哪一个策略
        if(symbol.equals(ADD_SYMBOL)){
            context = new Context(new Add());
        }else if(symbol.equals(SUB_SYMBOL)){
            context = new Context(new Sub());
        }else if(symbol.equals(MUL_SYMBO)){
            context = new Context(new Mul());
        }
        System.out.println("运行结果为:"+a+symbol+b+"="+context.exec(a,b,symbol));
    }
}

我们总结下这样的写的优点:

  • 算法可以自由切换
    这是策略模式本身定义的,只要实现抽象策略,它就成为策略家族的一个成员,通过封
    装角色对其进行封装,保证对外提供“可自由切换”的策略。
  • 避免使用多重条件判断
    如果没有策略模式,我们想想看会是什么样子?一个策略家族有5个策略算法,一会要
    使用A策略,一会要使用B策略,怎么设计呢?使用多重的条件语句?多重条件语句不易维
    护,而且出错的概率大大增强。使用策略模式后,可以由其他模块决定采用何种策略,策略
    家族对外提供的访问接口就是封装类,简化了操作,同时避免了条件语句判断。
  • 扩展性良好
    这甚至都不用说是它的优点,因为它太明显了。在现有的系统中增加一个策略太容易
    了,只要实现接口就可以了,其他都不用修改,类似于一个可反复拆卸的插件,这大大地符合了OCP原则。

当然他也不可避免的具有缺点:

  • 策略类数量增多
    每一个策略都是一个类,复用的可能性很小,类数量增多。
  • 所有的策略类都需要对外暴露
    上层模块必须知道有哪些策略,然后才能决定使用哪一个策略,这与迪米特法则是相违
    背的,我只是想使用了一个策略,我凭什么就要了解这个策略呢?那要你的封装类还有什么
    意义?这是原装策略模式的一个缺点,幸运的是,我们可以使用其他模式来修正这个缺陷,
    如工厂方法模式、代理模式或享元模式。

那我们什么应该使用策略模式呢:

  • 多个类只有在算法或行为上稍有不同的场景。
  • 算法需要自由切换的场景。
    例如,算法的选择是由使用者决定的,或者算法始终在进化,特别是一些站在技术前沿
    的行业,连业务专家都无法给你保证这样的系统规则能够存在多长时间,在这种情况下策略
    模式是你最好的助手。
  • 需要屏蔽算法规则的场景。
    现在的科技发展得很快,人脑的记忆是有限的(就目前来说是有限的),太多的算法你
    只要知道一个名字就可以了,传递相关的数字进来,反馈一个运算结果,万事大吉。

Android中有一个需求场景是不是特别像?有数据的时候,要展示数据;无网络的时候,展示重试界面。对的,就是状态策略,根据不同的状态选取不同的策略,但是我们一般不单独使用策略模式,而是使用工厂方法来实现策略类的声明。也就是利用混编,扬长避短,达到最优的设计。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容

  • 【学习难度:★☆☆☆☆,使用频率:★★★★☆】直接出处:策略模式梳理和学习:https://github.com/...
    BruceOuyang阅读 1,444评论 3 5
  • 工厂模式类似于现实生活中的工厂可以产生大量相似的商品,去做同样的事情,实现同样的效果;这时候需要使用工厂模式。简单...
    舟渔行舟阅读 7,727评论 2 17
  • javascript设计模式与开发实践 设计模式 每个设计模式我们需要从三点问题入手: 定义 作用 用法与实现 单...
    穿牛仔裤的蚊子阅读 4,045评论 0 13
  • 1.初识策略模式 定义一系列的算法,把它们一个个封装起来,并且使它们可相互替换。本模式使得算法可独立于使用它的客户...
    王侦阅读 1,450评论 0 3
  • 今天休息。这是把清明小长假提前休了一天。也好,分散休息,不至于一下子歇得无聊。 原本打算出去踏踏青,感受一下春光。...
    阳觅阅读 157评论 0 2