首页 > 学院 > 开发设计 > 正文

设计模式---策略模式

2019-11-14 23:29:05
字体:
来源:转载
供稿:网友
设计模式---策略模式

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

  • 抽象策略角色: 策略类,通常由一个接口或者抽象类实现。
  • 具体策略角色:包装了相关的算法和行为。
  • 环境角色:持有一个策略类的引用,最终给客户端调用。

适用范围:

• 许多相关的类仅仅是行为有异。 “策略”提供了一种用多个行为中的一个行为来配置一个类的方法。即一个系统需要动态地在几种算法中选择一种。• 需要使用一个算法的不同变体。例如,你可能会定义一些反映不同的空间 /时间权衡的算法。当这些变体实现为一个算法的类层次时 ,可以使用策略模式。• 算法使用客户不应该知道的数据。可使用策略模式以避免暴露复杂的、与算法相关的数据结构。• 一个类定义了多种行为 , 并且这些行为在这个类的操作中以多个条件语句的形式出现。将相关的条件分支移入它们各自的Strategy类中以代替这些条件语句。

下面通过一个简单的案列讲解

例:

刘备要到江东娶老婆了,走之前诸葛亮给赵云(伴郎)三个锦囊妙计,说是按天机拆开解决棘手问题,嘿,还别说,真是解决了大问题,搞到最后是周瑜陪了夫人又折兵呀,那咱们先看看这个场景是什么样子的。先说这个场景中的要素:三个妙计,一个锦囊,一个赵云,妙计是小亮同志给的,妙计是放置在锦囊里,俗称就是锦囊妙计嘛,那赵云就是一个干活的人,从锦囊中取出妙计,执行,然后获胜,用 java 程序怎么表现这个呢?我们先看类图:

三个妙计是同一类型的东东,那咱就写个接口:

package com.oumyye.策略模式;/*** @author * I'm glad to share my knowledge with you all.* 首先定一个策略接口,这是诸葛亮老人家给赵云的三个锦囊妙计的接口**/public interface IStrategy { //每个锦囊妙计都是一个可执行的算法public void Operate();}

然后再写三个实现类,有三个妙计嘛:

one

package com.oumyye.策略模式;/*** @author * I'm glad to share my knowledge with you all.* 找乔国老帮忙,使孙权不能杀刘备*/public class BackDoor implements IStrategy {public void operate() { System. out.PRintln(" 找乔国老帮忙,让吴国太给孙权施加压力"); }}

two

package com.oumyye.策略模式;/*** @author * I'm glad to share my knowledge with you all.* 求吴国太开个绿灯*/public class GivenGreenLight implements IStrategy {public void operate() { System. out.println(" 求吴国太开个绿灯, 放行! "); }}

three

package com.oumyye.策略模式;/*** @author* I'm glad to share my knowledge with you all.* 孙夫人断后,挡住追兵*/public class BlockEnemy implements IStrategy {public void operate() { System. out.println(" 孙夫人断后,挡住追兵"); }}

好了,大家看看,三个妙计是有了,那需要有个地方放这些妙计呀,放锦囊呀:

package com.oumyye.策略模式;/*** @author * I'm glad to share my knowledge with you all.* 计谋有了,那还要有锦囊*/public class Context {//构造函数,你要使用那个妙计private IStrategy straegy;public Context(IStrategy strategy){ this. straegy = strategy; }//使用计谋了,看我出招了public void operate(){ this. straegy.operate(); }}

然后就是赵云雄赳赳的揣着三个锦囊,拉着已步入老年行列的、还想着娶纯情少女的、色迷迷的刘老爷子去入赘了,嗨,还别说,小亮的三个妙计还真是不错,瞅瞅:

package com.oumyye.策略模式;/** * @author  */public class ZhaoYun {    /**     * 赵云出场了,他根据诸葛亮给他的交代,依次拆开妙计     */    public static void main(String[] args) {        Context context;        // 刚刚到吴国的时候拆第一个        System.out.println("-----------刚刚到吴国的时候拆第一个-------------");        context = new Context(new BackDoor()); // 拿到妙计        context.operate(); // 拆开执行        System.out.println("/n/n/n/n/n/n/n/n");        // 刘备乐不思蜀了,拆第二个了        System.out.println("-----------刘备乐不思蜀了,拆第二个了-------------");        context = new Context(new GivenGreenLight());        context.operate(); // 执行了第二个锦囊了        System.out.println("/n/n/n/n/n/n/n/n");        // 孙权的小兵追了,咋办?拆第三个        System.out.println("-----------孙权的小兵追了,咋办?拆第三个-------------");        context = new Context(new BlockEnemy());        context.operate(); // 孙夫人退兵        System.out.println("/n/n/n/n/n/n/n/n");        /*         * 问题来了:赵云实际不知道是那个策略呀,他只知道拆第一个锦囊,而不知道是BackDoor这个妙计,咋办?         * 似乎这个策略模式已经把计谋名称写出来了         *          * 错! BackDoor、 GivenGreenLight、 BlockEnemy只是一个代码,你写成first、 second、         * third,没人会说你错!         *          * 策略模式的好处就是:体现了高内聚低耦合的特性呀,缺点嘛,这个那个,我回去再查查         */    }}

就这三招,搞的周郎是“陪了夫人又折兵”呀!这就是策略模式,高内聚低耦合的特点也表现出来了,还有一个就是扩展性,也就是 OCP 原则,策略类可以继续增加下去,只要修改 Context.java 就可以了,这个不多说了,自己领会吧。

策略模式的优点:
  • 相关算法系列Strategy类层次为Context定义了一系列的可供重用的算法或行为。继承有助于析取出这些算法中的公共功能。
  • 提供了可以替换继承关系的办法: 继承提供了另一种支持多种算法或行为的方法。你可以直接生成一个Context类的子类,从而给它以不同的行为。但这会将行为硬行编制到 Context中,而将算法的实现与Context的实现混合起来,从而使Context难以理解、难以维护和难以扩展,而且还不能动态地改变算法。最后你得到一堆相关的类 , 它们之间的唯一差别是它们所使用的算法或行为。将算法封装在独立的Strategy类中使得你可以独立于其Context改变它,使它易于切换、易于理解、易于扩展。
  • 消除了一些if else条件语句:Strategy模式提供了用条件语句选择所需的行为以外的另一种选择。当不同的行为堆砌在一个类中时 ,很难避免使用条件语句来选择合适的行为。将行为封装在一个个独立的Strategy类中消除了这些条件语句。含有许多条件语句的代码通常意味着需要使用Strategy模式。
  • 实现的选择Strategy模式可以提供相同行为的不同实现。客户可以根据不同时间 /空间权衡取舍要求从不同策略中进行选择。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表