扫盲:策略模式,成事儿还需要策略

什么是策略模式?

生活中的策略

策略模式在生活中体现很多。

我们要去旅游,我们可以选择不同的出行方式:飞机,火车,大巴,自驾等,这是不同的策略。

双十一当当网购买满减活动,满 100 减 50,满 200 减 100,满 400 减 250 等,这也是不同的策略。

抑或是我们在追求女生时,针对不同性格的女孩子采用不同的方式,这还是不同的策略。

程序中的策略

策略模式在程序中的体现依然淋漓尽致。

比如我们的图片加载,Android 上有 FrescoPicassoGlideUniversal-Image-Loader 等,iOS 上有 SDWebImageAFNetworkingFastImageCache 等。

所以,假设让你来设计一个图片加载上层框架,要求可以底层可以使用 A B 两种加载策略,你会怎么做呢?

// 加载类A 
public class ImageLoadServiceA {
    public void loadImage() {
        System.out.println("使用 A 加载框架");
    }
}
// 加载类B
public class ImageLoadServiceB {
    public void loadImage() {
        System.out.println("使用 B 加载框架");
    }
}

// 使用
public void loadNetImage(boolean useA) {
    if(useA){
        new ImageLoadServiceA().loadImage();// 使用A加载方式  
    } else {
        new ImageLoadServiceB().loadImage();// 使用B加载方式
    }
}

可以看到,上述通过一个 useA 参数判断是否使用 A 框架,为 true 使用 A,否则使用 B 框架进行加载。

使用简单工厂模式应对

但假设我们现在需要再支持一个 C 框架的使用,你可能想到了,那就再加一个 boolean 参数 useB 即可,或者直接使用一个 int 参数 loadType,宏定义 0 代表 A 框架,1 代表 B 框架,2 代表 C 框架,这样如果需要增加方式则更新取值即可。

设计模式不过是我们写程序的招式,由于之前大家可能还学习过了简单工厂模式,我们不妨在这里进行实战。

// 抽象图片加载类 
public abstract class ImageLoadService {
    public abstract void loadImage();
}       

// 具体加载类A 
public class ImageLoadServiceA extends ImageLoadService {
    @Override
    public void loadImage() {
        System.out.println("使用 A 加载框架");
    }
}
//具体加载类B
public class ImageLoadServiceB extends ImageLoadService {
    @Override
    public void loadImage() {
        System.out.println("使用 B 加载框架");
    }
}

//具体加载类C
public class ImageLoadServiceC extends ImageLoadService {
    @Override
    public void loadImage() {
        System.out.println("使用 C 加载框架");
    }
}

public class ImageLoadFactory {
    public static ImageLoadService create(int loadType) {
        ImageLoadService loadService = null;
        switch (loadType) {
            case 0:
                loadService = new ImageLoadServiceA();
                break;
            case 1:
                loadService = new ImageLoadServiceB();
                break;
            case 2:
                loadService = new ImageLoadServiceC();
                break;
        }
        return loadService;
    }
}

// 使用
public void loadNetImage(int loadType) {
    ImageLoadFactory.create(loadType).loadImage();
}

可以看到,我们使用简单工厂模式后,在处理新增其他加载方式的问题的时候,不会再去影响原有的加载类代码,如果新增一种加载方式的话,我们只需要新增 ImageLoadXXX 类,实现 loadImage() 加载方法,再修改工厂类 ImageLoadFactory 即可。

相信你也发现了,这个方式只能解决对象的创建问题,我们每次新增方式的时候都会新增一个类,而且需要对工厂类进行代码修改,显然是违反了开闭原则。

策略模式

人生处处有策略,上面的不同的加载方式其实就是不同的「策略」。

策略模式是对 算法的封装,它将每一个算法封装到具有共同接口的独立的类中,从而使得它们可以独立变换。

策略模式的特点

  • 是一种行为模式,对算法封装,使得客户端独立于各个策略;
  • 扩展性强,添加策略无非就是添加一个具体的实现类而已,代价非常低;

策略模式的结构


类图

策略模式做实现

要学习一个设计模式,先要学会临摹,所以上面的需求,我们可以实现为:

  1. 定义抽象策略
public interface ImageLoadStrategy {
    void loadImage() ;
}
  1. 定义具体的策略
// 具体加载类A 
public class ImageLoadStrategyA implements ImageLoadStrategy {
    @Override
    public void loadImage() {
        System.out.println("使用 A 加载框架");
    }
}
//具体加载类B
public class ImageLoadStrategyB implements ImageLoadStrategy {
    @Override
    public void loadImage() {
        System.out.println("使用 B 加载框架");
    }
}

//具体加载类C
public class ImageLoadStrategyC implements ImageLoadStrategy {
    @Override
    public void loadImage() {
        System.out.println("使用 C 加载框架");
    }
}
  1. 定义上下文,选择方式
public class ContextImageLoadStrategy {

    private ImageLoadStrategy strategy ;

    public ContextImageLoadStrategy(ImageLoadStrategy strategy){
        this.strategy = strategy ;
    }

    public void loadImage(){
        strategy.loadImage();
    }
}
  1. 使用
public void loadImage(ImageLoadStrategy imageLoadStrategy){
    ContextImageLoadStrategy contextStrategy = new ContextImageLoadStrategy(imageLoadStrategy);
    contextStrategy.loadImage();
}   

注意: 策略的核心不是如何实现算法,而是如何更优雅的把这些算法组织起来,让客户端非常好调用「虽然策略非常多,可以自由切换,但是同一时间客户端只能调用一个策略,其实也很好理解,你不可能同时既坐飞机,又坐火车」。

策略模式的优点

  • 策略类可以互相替换
    由于策略类都实现同一个接口,因此他们能够互相替换。
  • 耦合度低,方便扩展
    增加一个新的策略只需要添加一个具体的策略类即可,基本不需要改变原有的代码,符合开闭原则。
  • 避免使用多重条件选择语句(if-else 或者 switch)。

策略模式的缺点

  • 策略的增多会导致子类的也会变多。比如上方再增加加载方式必须增加类。
  • 客户端必须知道所有的策略类,并自行决定使用哪一个策略类。比如上方必须知道有哪些加载策略,这样我们才能调用到正确的加载方式。

你有想到如何解决「客户端必须知道所有的策略类」这个缺点么?

策略模式的应用场景

  • 同一个问题具有不同算法时,即仅仅是具体的实现细节不同时,如各种排序算法等等。
  • 对客户隐藏具体策略(算法)的实现细节,彼此完全独立;提高算法的保密性与安全性。
  • 一个类拥有很多行为,而又需要使用 if-else 或者 switch 语句来选择具体行为时。使用策略模式把这些行为独立到具体的策略类中,可以避免多重选择的结构。

源码中的策略模式

想必大家已经很清楚上面的策略模式了,下面源码中用到策略模式了吗?

  • Android 的动画插值器;
  • Android 中 ListViewArrayAdapterSimpleAdapter

写在最后

总的来说,策略模式还算我们项目开发中会使用非常频繁的模式,你学会了么?如有疑问,请在评论区留言。

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

推荐阅读更多精彩内容