Java设计模式之策略模式

Java设计模式之策略模式

简介

在策略模式(Strategy Pattern)中,一个类的行为或其算法可以在运行时更改。这种类型的设计模式属于行为型模式。
在策略模式中,我们创建表示各种策略的对象和一个行为随着策略对象改变而改变的 context 对象。策略对象改变 context 对象的执行算法。

代码演示

参考<Head First设计模式>这本书中的demo,实现代码

  • 该类是鸭子的基类,想象中所有的鸭子都会有飞行,叫声,游泳三种行为,接下来我们来实现三种类型的鸭子绿头鸭,红头鸭,以及石头刻的鸭子
public abstract class BaseDuck {
    public BaseDuck() {
    }
    /**
     * 种类
     */
    public abstract void display();
    public void fly() {
        System.out.println("~~im fly~~");
    }
    public void quack() {
        System.out.println("~~gaga~~");
    }
    public void swim() {
        System.out.println("~~im swim~~");
    }
}

  • 先来绿头鸭

public class GreenHeadBaseDuck extends BaseDuck {
    @Override
    public void display() {
        System.out.println("**GreenHead**");
    }
}
  • 再来红头鸭
public class RedHeadBaseDuck extends BaseDuck {
    @Override
    public void display() {
        System.out.println("**RedHead**");
    }

}
  • 然后有一天,新的需求来了,现在需要一种用石头雕刻的鸭子,它不会游泳,不会叫,不会飞,如果按照之前的继承方法,石头鸭子居然会飞,会叫,会游泳了,神了奇了!!
public class StoneBaseDuck extends BaseDuck {
    @Override
    public void display() {
        System.out.println("~~stone duck~~");
    }
}

很明显这样是有问题的,该怎么解决呢,这时候就会想到我们可以重写基类的方法啊,让他不会叫不会飞,也不会游泳.好了问题看似解决了.
然后产品经理告诉你,现在需要鸭子的另一个性质--羽毛,然后我们在基类中加入羽毛属性,然后发现石头鸭子没有羽毛,接着我们又要去重写石头鸭子中的羽毛方法.....
接着一个又一个新属性被加上来,每天在新增修改,重写中疲于奔命....

这就是,** 继承的操作中对超类的局部改动,会影响继承他的子类,产生的溢出效应 **,软件开发中我们常说的开闭原则,对扩展开放,对修改关闭,
在现代项目中面对新的需求不断的增加,就需要提高代码中的可扩展性,降低复杂度,

  • 分析项目中变化和不变化的部分,提取变化的部分,抽象成接口+实现
  • 鸭子的那些属性是会变化的,将其分析提取出来

好了基于上面的思考,我们将上面的代码进行拆分重构,

  • 首先我们先新建飞行行为接口,后面新增任何飞行动作,实现飞行这个接口,完成飞行动作的扩展.
public interface FlyBehavior {

    /**
     * 飞行行为
     */
    void fly();
}
  • 飞行动作好的实现
public class GoodFlyBehavior implements FlyBehavior {

    @Override
    public void fly() {
        System.out.println("--GoodFly--");
    }

}
  • 飞行动作一般的实现
public class BadFlyBehavior implements FlyBehavior {
    @Override
    public void fly() {
        System.out.println("--BadFly--");
    }
}
  • 不会飞的实现
public class NoFlyBehavior implements FlyBehavior {
    @Override
    public void fly() {
        System.out.println("--NoFly--");
    }
}
  • 叫声接口,新增的叫声都实现该接口进行扩展
public interface QuackBehavior {
    /**
     * 鸭子叫声行为
     */
    void quack();
}
  • 嘎嘎叫的声音
public class GaGaQuackBehavior implements QuackBehavior {
    @Override
    public void quack() {
        System.out.println("__GaGa__");
    }
}
  • 咯咯的叫声
public class GeGeQuackBehavior implements QuackBehavior {
    @Override
    public void quack() {
        System.out.println("__GeGe__");
    }
}
  • 不会叫

public class NoQuackBehavior implements QuackBehavior {
    @Override
    public void quack() {
        System.out.println("__No Quack__");
    }
}
  • 游泳的动作接口,跟上面两个接口一样
public interface SwimBehavior {
    void swim();
}
  • 游泳技术非常好
public class GoodSwimBehavior implements SwimBehavior {
    @Override
    public void swim() {
        System.out.println("~~Good swim~~");
    }
}
  • 游泳技术一般
public class BadSwimBehavior implements SwimBehavior {
    @Override
    public void swim() {
        System.out.println("~~bad swim~~");
    }
}
  • 不会游泳
public class NoSwimBehavior implements SwimBehavior {
    @Override
    public void swim() {
        System.out.println("~~no swim~~");
    }
}
  • BaseDuck基类,在基类中调用之前的接口,完成相应的行为
public abstract class BaseDuck {
    /**
     * 飞行动作接口
     */
    FlyBehavior mFlyBehavior;
    /**
     * 鸭子叫声接口
     */
    QuackBehavior mQuackBehavior;
    /**
     * 游泳接口
     */
    SwimBehavior mSwimBehavior;
    
    public BaseDuck() {
    }
    /**
     * 种类
     */
    public abstract void display();
    
    public void fly() {
        mFlyBehavior.fly();
    }
    /**
     * 设置新的飞行行为
     * @param flyBehavior
     */
    public void setFlyBehavior(FlyBehavior flyBehavior) {
        mFlyBehavior = flyBehavior;
    }
    public void quack() {
        mQuackBehavior.quack();
    }
    /**
     * 设置新的叫声属性
     * @param quackBehavior
     */
    public void setQuackBehavior(QuackBehavior quackBehavior) {
        mQuackBehavior = quackBehavior;
    }
    public void swim() {
        mSwimBehavior.swim();
    }
}
  • 绿头鸭品种,绿头鸭中飞行行为一般,会嘎嘎叫,游泳非常好,通过绿头鸭的构造方法自定义属性
public class GreenHeadBaseDuck extends BaseDuck {
    public GreenHeadBaseDuck(){
        mFlyBehavior = new BadFlyBehavior();
        mQuackBehavior = new GaGaQuackBehavior();
        mSwimBehavior = new GoodSwimBehavior();
    }
    @Override
    public void display() {
        System.out.println("**GreenHead**");
    }
}
  • 红头鸭的品种
public class RedHeadBaseDuck extends BaseDuck {
    public RedHeadBaseDuck() {
        mFlyBehavior = new GoodFlyBehavior();
        mQuackBehavior = new GeGeQuackBehavior();
        mSwimBehavior = new BadSwimBehavior();
    }
    @Override
    public void display() {
        System.out.println("**RedHead**");
    }
}
  • 石头做的鸭子,实现自定义属性
public class StoneBaseDuck extends BaseDuck {
    public StoneBaseDuck(){
        mFlyBehavior = new NoFlyBehavior();
        mQuackBehavior = new NoQuackBehavior();
        mSwimBehavior = new NoSwimBehavior();
    }
    @Override
    public void display() {
        System.out.println("**StoneBaseDuck**");
    }
}
  • 测试方法
public class StimulateDuck {

    public static void main(String[] args) {

        BaseDuck mGreenHeadBaseDuck = new GreenHeadBaseDuck();
        BaseDuck mRedHeadBaseDuck = new RedHeadBaseDuck();
        BaseDuck stoneBaseDuck = new StoneBaseDuck();

        mGreenHeadBaseDuck.display();
        mGreenHeadBaseDuck.fly();
        mGreenHeadBaseDuck.quack();
        mGreenHeadBaseDuck.swim();

        mRedHeadBaseDuck.display();
        mRedHeadBaseDuck.fly();
        mRedHeadBaseDuck.quack();
        mRedHeadBaseDuck.swim();
        System.out.println("==修改红头鸭属性==");
        mRedHeadBaseDuck.display();
        mRedHeadBaseDuck.setFlyBehavior(new NoFlyBehavior());
        mRedHeadBaseDuck.fly();
        mRedHeadBaseDuck.setQuackBehavior(new NoQuackBehavior());
        mRedHeadBaseDuck.quack();

        stoneBaseDuck.display();
        stoneBaseDuck.fly();
        stoneBaseDuck.quack();
    }

}

  • 打印结果
**GreenHead**
--BadFly--
__GaGa__
~~Good swim~~
**RedHead**
--GoodFly--
__GeGe__
~~bad swim~~
==修改红头鸭属性==
**RedHead**
--NoFly--
__No Quack__
**StoneBaseDuck**
--NoFly--
__No Quack__

这样一个策略模式的就完成了,所有的新增行为动作都不会修改原有的代码,只用在原来的基础上进行扩展,符合了开闭原则

总结

策略模式的实现就是分别对行为封装接口,实现算法族,基类里面放置行为接口对象,在子类里具体设定行为对象,
原则就是,分离变化部分,封装接口,基于接口编程各种功能,此模式使得行为算法的变化独立于算法使用者.

参考代码链接

Gitee: https://gitee.com/singlekingdom/JavaDesignPatterns.git

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