Android 策略模式

源码地址

介绍

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

使用场景

  • 针对同一类型的多种处理方式,仅仅是具体行为有差别时。
  • 需要安全地封装多种同一类型的操作时。
  • 出现同一抽象类有多个子类,而又需要使用 if-else 或者 switch-case 来选择具体子类时。

角色介绍

  • Context——用来操作策略的上下文环境;
  • Strategy——策略的抽象;
  • ConcreteStrategyA,ConcreteStrategyB——具体的策略实现。

简单实现

需求:下面以坐公交工具的费用计算来演示。

第一个版本的代码:

public class PriceCalculator {
    //公交车类型
    private static final int BUS = 1;
    //地铁类型
    private static final int SUBWAY = 2;

    /**
     * 公交车,十公里之内一元钱,超过十公里之后每加一元钱可以乘5公里
     * @param km   公里
     * @return
     */
    private int busPrice(int km) {
        //超过十公里的总距离
        int extraTotal = km - 10;
        //超过的距离是5公里的倍数
        int extraFactor = extraTotal / 5;
        //超过的距离对5公里取余
        int fraction = extraTotal % 5;
        //价格计算
        int price = 1 + extraFactor * 1;
        return fraction > 0 ? ++price : price;
    }

    /**
     * 6公里(含)内3元;6~12公里(含)4元;12~22公里(含)5元;22~32公里(含)6元;
     * @param km    公里
     * @return
     */
    private int subwayPrice(int km) {
        if (km <= 6) {
            return 3;
        } else if (km <= 12) {
            return 4;
        } else if (km <= 22) {
            return 5;
        } else if (km <= 32) {
            return 6;
        }
        //其他距离简化为7元
        return 7;
    }

    private int calculatePrice(int km, int type) {
        if (type == BUS) {
            return busPrice(km);
        } else if (type == SUBWAY) {
            return subwayPrice(km);
        }
        return 0;
    }

    public static void main(String[] args) {
        PriceCalculator calculator = new PriceCalculator();
        System.out.println("坐16公里的公交车票价为:" + calculator.calculatePrice(16, BUS));
        System.out.println("坐16公里的地铁票价为:" + calculator.calculatePrice(16, SUBWAY));
    }
}

当增加一种出行方式时,如出租车,那么就需要在 PriceCalculator 中增加一个方法来计算出租车出行的价格,并且在 calculatePrice 函数中增加一个判断,新增代码大致如下:

public class PriceCalculator {
    //公交车类型
    private static final int BUS = 1;
    //地铁类型
    private static final int SUBWAY = 2;
    //出租车类型
    private static final int TAXI = 3;

    private int taxiPrice(int km) {
        return km * 2;
    }
  
    private int calculatePrice(int km, int type) {
        if (type == BUS) {
            return busPrice(km);
        } else if (type == SUBWAY) {
            return subwayPrice(km);
        } else if (type == TAXI) {
            return taxiPrice(km);
        }
        return 0;
    }
}

当修改计算方式或者增加出行方式时,有需要增加 if-else 代码,这样会使得代码变得难以维护。

下面为策略模式实现代码:

  1. 计算接口

    public interface CalculateStrategy {
        /**
         * 按距离来计算价格
         * @param km    公里
         * @return  返回价格
         */
        int calculatePrice(int km);
    }
    
  2. 公交车价格计算策略

    public class BusStrategy implements CalculateStrategy {
        /**
         * 公交车,十公里之内一元钱,超过十公里之后每加一元钱可以乘5公里
         * @param km   公里
         * @return
         */
        @Override
        public int calculatePrice(int km) {
            //超过十公里的总距离
            int extraTotal = km - 10;
            //超过的距离是5公里的倍数
            int extraFactor = extraTotal / 5;
            //超过的距离对5公里取余
            int fraction = extraTotal % 5;
            //价格计算
            int price = 1 + extraFactor * 1;
            return fraction > 0 ? ++price : price;
        }
    }
    
  3. 地铁价格计算策略

    public class SubwayStrategy implements CalculateStrategy {
        /**
         * 6公里(含)内3元;6~12公里(含)4元;12~22公里(含)5元;22~32公里(含)6元;
         * @param km    公里
         * @return
         */
        @Override
        public int calculatePrice(int km) {
            if (km <= 6) {
                return 3;
            } else if (km <= 12) {
                return 4;
            } else if (km <= 22) {
                return 5;
            } else if (km <= 32) {
                return 6;
            }
            //其他距离简化为7元
            return 7;
        }
    }
    
  4. 出行计算器

    public class TranficCalculator {
        private CalculateStrategy mStrategy;
        public void setStrategy(CalculateStrategy strategy) {
            mStrategy = strategy;
        }
    
        public int calculatePrice(int km) {
            return mStrategy.calculatePrice(km);
        }
    
        public static void main(String[] args) {
            TranficCalculator calculator = new TranficCalculator();
            calculator.setStrategy(new BusStrategy());
            System.out.println("公交车乘坐16公里的价格:" + calculator.calculatePrice(16));
    
            calculator.setStrategy(new SubwayStrategy());
            System.out.println("地铁乘坐16公里的价格:" + calculator.calculatePrice(16));
    }
    

    如果需要增加一个出租车计算策略类,代码如下:

  5. 出租车计算策略

    public class TaxiStrategy implements CalculateStrategy {
        @Override
        public int calculatePrice(int km) {
            return km * 2;
        }
    }
    
  6. 出行价格计算器

    public class TranficCalculator {
        private CalculateStrategy mStrategy;
        public void setStrategy(CalculateStrategy strategy) {
            mStrategy = strategy;
        }
    
        public int calculatePrice(int km) {
            return mStrategy.calculatePrice(km);
        }
    
        public static void main(String[] args) {
            TranficCalculator calculator = new TranficCalculator();
            calculator.setStrategy(new TaxiStrategy());
            System.out.println("出租车乘坐16公里的价格:" + calculator.calculatePrice(16));
        }
    }
    

从上面可以看出,使用 if-else 来解决问题,优点是实现简单,层级单一;但缺点也很明显,即代码臃肿,逻辑复杂,难以升级与维护,没有结构可言。

后者是通过建立抽象,将不同的策略构建成一个具体的策略实现,通过不同的策略实现算法替换。在简化逻辑、结构的同时,增强了系统的可读性、稳定性、可扩展性。

Android源码中的实现

TimeInterpolator 时间插值器,用在属性动画中。

总结

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

推荐阅读更多精彩内容

  • 1 场景问题# 1.1 报价管理## 向客户报价,对于销售部门的人来讲,这是一个非常重大、非常复杂的问题,对不同的...
    七寸知架构阅读 5,035评论 9 62
  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,723评论 0 33
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,596评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,571评论 18 399
  • 关于作者 钱钟书,中国现代文学史上非常重要的一位“才子型”文学家。他1933年从清华大学外文系毕业,作为最早的庚子...
    蔚成阅读 245评论 0 0