流行框架源码分析(15)-Strategy策略模式

主目录见:Android高级进阶知识(这是总目录索引)
 策略模式应该说应用也是非常广泛,而且很容易使用。有的人可能用到了但是没有意识到,那么我们今天会让大家意识到而且能在特定的场景中使用到他。我们这里先来看看他的定义:

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

这里的算法意思不是传统的算法,其实就是一个action,就是一个独立的执行逻辑。策略模式可以独立出来这些逻辑,然后在特定场景选择特定的执行逻辑。接着我们来看看他的UML类图:


策略模式

角色介绍:

  • IStrategy:策略接口,定义了各个策略的公共接口
  • ConcreteStrategy:具体的策略类,里面实现了策略接口中的策略方法。
  • Context:用来操作各个策略的,利用具体的策略类进行各种策略执行。

一.目标

我们今天的目标跟之前讲的设计模式也是一样,我们在明白别人使用的场景的时候自己遇到的时候也能利用起来。所以今天目标也同样是:
1.学会策略模式在什么场景下使用;
2.能在自己编写框架的时候用上这个设计模式。

二.模式解析

首先也是我们这里先假设一个场景:我们大学编程的时候,遇到一个编程题目解不出来,那么这时候有几种情况可以解决。一个是查看答案,一个是求教同学,一个是请教老师。那我们平常写代码可能是这样:

public class StrategyPatternMain {
    
    enum Strategy{
        CHECK_ANSWER,SEARCH_HELP,CONSULT_TEACHER
    }
    
    private Strategy strategy;
    public StrategyPatternMain(Strategy strategy){
        super();
        this.strategy = strategy;
    }
    
    public void doAction(){
        switch (strategy) {
        case CHECK_ANSWER:
            System.out.println("查看答案");
            break;
        case SEARCH_HELP:
            System.out.println("请教同学");
            break;
        case CONSULT_TEACHER:
            System.out.println("请教老师");
            break;
        default:
            break;
        }
    }

    public static void main(String[] args) {
        StrategyPatternMain homeWorkStrategy = new StrategyPatternMain(Strategy.SEARCH_HELP);
        homeWorkStrategy.doAction();
        
        StrategyPatternMain homeWorkStrategy2 = new StrategyPatternMain(Strategy.CHECK_ANSWER);
        homeWorkStrategy2.doAction();
        
        StrategyPatternMain homeWorkStrategy3 = new StrategyPatternMain(Strategy.CONSULT_TEACHER);
        homeWorkStrategy3.doAction();

    }
}

可以看到,完美有没有,你来一个新的方式我这边可以再增加一个case,灵活有没有。然后最后你想到方法越来越多,这个类就被修改了无数遍,代码就是一坨一坨的,这时候你想,不行呀,我应该让他可以组装呀,而且不影响原有的代码。那么策略模式就蹦出来了。

1.策略模式

我们这里解决这个扩展困难问题,我们这里引入策略模式,首先我们先来看看策略接口:

public interface IStrategy {
    void doAction();
}

这个策略接口类很简单,就是一个方法,然后我们来看看几个具体策略类:

public class CheckAnswer implements IStrategy {

    @Override
    public void doAction() {
        System.out.println("查看答案");
    }
}

public class SearchHelp implements IStrategy {

    @Override
    public void doAction() {
        System.out.println("请教同学");
    }
}

public class ConsultTeacher implements IStrategy {

    @Override
    public void doAction() {
        System.out.println("请教老师");
    }
}

我们看到几个策略类也是非常简单的,就是简单地执行各自的打印动作。然后我们看下Context类:

public class Context {
    private IStrategy strategy;
    
    public void setStrategy(IStrategy strategy){
        this.strategy = strategy;
    }
    
    public void doAction(){
        strategy.doAction();
    }
}

我们看到这里面持有了一个策略类的引用,然后在方法里面调用相应的action。我们看下这个怎么使用:

        IStrategy checkAnswer = new CheckAnswer();
        IStrategy searchHelp = new SearchHelp();
        IStrategy consultTeacher = new ConsultTeacher();
        
        Context context = new Context();
        context.setStrategy(checkAnswer);
        context.doAction();
        
        context.setStrategy(searchHelp);
        context.doAction();
        
        context.setStrategy(consultTeacher);
        context.doAction();

我们看到这边如果增加了寻找解决问题的方法,我们只要增加一个具体策略类,然后在使用的时候,设置进去就可以了,这样的话面对扩展就非常地方便。这也是我们为啥要使用这个模式的原因。

2.属性动画中策略模式的使用

属性动画中的插值器(Interpolator)和估值器(Evaluator)的使用就是用的策略模式,我们先来看下插值器。我们先来看下之前写过一篇[属性动画基础用法]中有个属性用法是这样的:

  Point endPoint = new Point(getWidth() - RADIUS,getHeight() - RADIUS);
        ValueAnimator valueAnimator = ValueAnimator.ofObject(new PointEvaluator(),currentPoint,endPoint);
        valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator valueAnimator) {
                currentPoint = (Point) valueAnimator.getAnimatedValue();
                invalidate();
            }
        });
//这里设置了插值器
        valueAnimator.setInterpolator(new AccelerateDecelerateInterpolator());
        valueAnimator.setDuration(2000);
        valueAnimator.start();

我们看到这里设置了具体的插值器,那么我们来分析下这里面的策略模式的各个 角色,首先就是Context角色,这个很明显就是ValueAnimator,我们看下他的setInterpolator方法:

  @Override
    public void setInterpolator(TimeInterpolator value) {
        if (value != null) {
            mInterpolator = value;
        } else {
            mInterpolator = new LinearInterpolator();
        }
    }

可以看到这个里面设置了具体插值器的引用。然后我们看看具体插值器也就是具体策略类的实现:

@HasNativeInterpolator
public class LinearInterpolator extends BaseInterpolator implements NativeInterpolatorFactory {

    public LinearInterpolator() {
    }

    public LinearInterpolator(Context context, AttributeSet attrs) {
    }

    public float getInterpolation(float input) {
        return input;
    }

    /** @hide */
    @Override
    public long createNativeInterpolator() {
        return NativeInterpolatorFactoryHelper.createLinearInterpolator();
    }
}

我们看到这个具体插值器继承了BaseInterpolator 并且实现了NativeInterpolatorFactory。这里的BaseInterpolator 实际又实现了Interpolator,然后Interpolator又实现了TimeInterpolator:

public interface TimeInterpolator {
    float getInterpolation(float input);
}

所以TimeInterpolator 我们这里可以当做抽象插值器类也就是抽象策略类。看到这里我们已经把每个策略模式的角色都分出来了。其实是非常简单的,但是达到的效果确实符合程序优秀设计的思想。至于估值器其实也是类似的,这里就不举那么多例子了,大家看懂就可以。其实在Glide里面大家设置缓存策略的时候也会用到策略设计模式,大家有兴趣可以自己也看看。
总结:策略模式使用简单,场景也非常多,记住遇到if-else,switch特别多的场景,同时程序后面可能会变化,条件会增加。这时候往往就是策略模式派上用场的时候,所以我们平常用设计模式的时候要多想多用。

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,579评论 18 139
  • 本文首发于个人博客:Lam's Blog - 谈谈23种设计模式在Android源码及项目中的应用,文章由Mark...
    格子林ll阅读 4,615评论 1 105
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,368评论 25 707
  • 我有个好朋友,她算是个活在梦幻里的人,喜欢笑,敢于哭,爱热闹。高三时候,她和一个男生交往。那个男生是她之前最...
    金污一阅读 261评论 0 0
  • 这幅画大家一定很陌生,容喜欢历史的我介绍一下: 新郎是光绪皇帝,清朝的第十一位皇帝;新娘是叶赫那拉·静芬,...
    帅梅香茗阅读 1,225评论 5 2