今天看啥  ›  专栏  ›  码农UP2U

设计模式之策略模式(Strategy Pattern)

码农UP2U  · 简书  ·  · 2019-11-09 10:29

模板方法是通过继承实现的,在父类中定义出算法的骨架,将不同点在子类中实现。而策略模式是通过接口实现的,策略中定义了完整的算法。它们有点像啊……

策略模式的定义

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

我的理解是,策略模式是透明的,对于策略的实现有时是不用关心的,我只要达到我的目的就可以了,而具体的实现可能有多种手段,而根据不同的选择可以有不同的手段。而对于模板方法来说,它的流程是基本固定的,而某个环境中的实现方式是可变的,但是整体是不变的。这是我理解的它们之间的不同,至于编码的话,模板方法使用的是继承的方式,而策略模式使用的是接口。

代码示例

仍然以吃面为主,比如程序员要吃面,程序员中午可能去便利店让店员泡面来吃,而程序员黑夜回家可能让老妈给煮面来吃。对于我们来说,我们是要吃面的,至于吃什么面,自己说一下就可以了,至于吃的这个面是怎么做的,就不关心了,做面的策略也根据不同的面有所不同。

class StrategyPattern {
    public static void main(String[] args) {
        Ren cxy = new ChengXuYuan();
        
        System.out.println("程序员中午吃饭");
        
        cxy.setIZm(new PaoMian());
        cxy.eat();
        
        System.out.println("程序员晚上吃饭");
        
        cxy.setIZm(new ZhuMian());
        cxy.eat();
    }
}

interface IZuoMian {
    public void zuoMian();
}

class PaoMian implements IZuoMian {
    public void zuoMian() {
        System.out.println("揭开碗面的包装纸");
        System.out.println("冲开水");
        System.out.println("泡面中...");
        System.out.println("泡好了");
    }
}

class ZhuMian implements IZuoMian {
    public void zuoMian() {
        System.out.println("烧开水...");
        System.out.println("下面...");
        System.out.println("捞面...");
    }
}

abstract class Ren {
    public IZuoMian zm;
    
    public void setIZm(IZuoMian zm) {
        this.zm = zm;
    }
    
    public void eat() {
        zm.zuoMian();
        System.out.println("吃面");
    }
}

class ChengXuYuan extends Ren {
    
}

在Ren类中,都是要有eat()方法的,是人就要吃!但是吃什么要自己说。根据自己要吃的不同,具体做的也就不同了,这里是做面吃,当然这个做面也可以是做饭。

在代码中实现了泡面和煮面两种方式,做面是一个接口即IZuoMian(),通过把做面的具体实现的类传递给Ren类中的接口属性,就可以通过该属性来完成具体的做面过程。

执行输出如下:


策略模式输出结果

总结:

总结来自于《大话设计模式》一书中第二章“商场促销——策略模式”。

1、面向对象的编程,并不是类越多越好,类的划分是为了封装,但分类的基础是抽象,具有相同属性和功能的对象的抽象集合才是类。(这里是我的理解:也就是说策略只是一个算法,可以说是有相应的功能,而没有相应的属性,因此封装为一个接口更为合适)
2、策略模式(Strategy):它定义了算法家族,分别封装起来,让它们之间可以相互替换,此模式让算法的变化,不会影响到使用算法的客户;
3、策略模式是一种定义一系列算法的方法,从概念上来看,所有这些算法完成的都是相同的工作,只是实现不同,它可以以相同的方式调用所有的算法,减少了各种算法类与使用算法类之间的耦合;
4、策略模式的Strategy类层次为Context定义了一系列的可供重用的算法或行为。继承有助于析取出这些算法中的公共功能;
5、策略模式的优点是简化了单元测试,因为每个算法都有自己的类,可以通过自己的接口单独测试;
6、当不同的行为堆切在一个类中时,就很难避免使用条件语句来选择合适的行为。将这些行为封装在一个个独立的Strategy类中,可以在使用这些行为的类中消除条件语句;
7、策略模式就是用来封装算法的,但在实践中,我们发现可以用它来封装几乎任何类型的规则,只要在分析过程中听到需要在不同时间应用不同的业务规则,就可以考虑使用策略模式处理这种变化的可能性;
8、在基本的策略模式中,选择所用具体实现的职责由客户端对象承担,并转给策略模式的Context对象。



我的微信公众号:“码农UP2U”
我的公众号




原文地址:访问原文地址
快照地址: 访问文章快照