设计模式之策略模式

简介: 设计模式之策略模式

定义一组算法,将每个算法都封装起来,并且使它们之间可以互换。策略模式使这些算法在客户端调用它们的时候能够互不影响地变化。(Java的TreeSet集合中,构造方法可传入具体的比较器对象以实现不同的排序算法。就是利用的策略模式)策略模式的用意是针对一组算法,将每一个算法封装到具有共同接口的独立的类中, 从而使得它们可以相互替换,使用策略模式可以把行为和环境分割开来。

角色

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

场景假设

模 拟 鸭 子 游 戏 :SimUDuck。游戏中会出现各种鸭子,一边游泳戏水,一边呱

呱叫。不同的鸭子有不同的行为,所以我们可以把鸭子会不会飞,怎么飞的行为定义为抽象策略角色,鸭子叫行为也定义成一种策略,因为叫法不同。

分开变与不变

把会变化的部分取出并封装起来,以便以后可以轻易地改动或扩充此部分,而不影响不需要变化的其他部分。

我们先抽象一个Duck类,鸭子有 fly() 和 quack() ,但是会随着鸭子的不同而改变,所以我们要抽出来,将他们定义为变的部分涉及成策略,Duck 类持有对应策略的引用,通过组合的方式实现。

image.png

设计类图

image.png

最后我们整合鸭子的行为,分别将鸭子的飞行与嘎嘎叫的动作 "委托"(delagate)别人处理,而不是定义在Duck类中。

我们用两个相似的方法performFly()和performQuack()取代鸭子类中的fly()与quack()。

代码实现

定义 Duck 抽象类,所有的鸭子都继承。通过组合代理的模式实现飞行与嘎嘎叫的功能,持有策略类(飞行,呱呱叫)的引用。

提供set方法指定不同的策略,然后通过 performFly 与 performQuack 委托对应的策略实现。

环境角色

public abstract class Duck {
    FlyBehavior flyBehavior;
    QuackBehavior quackBehavior;
    public void performFly() {
        flyBehavior.fly();
    }
    public void performQuack() {
        quackBehavior.quack();
    }
    /**
     * 展示鸭子
     */
    public abstract void display();
    public void setFlyBehavior(FlyBehavior flyBehavior) {
        this.flyBehavior = flyBehavior;
    }
    public void setQuackBehavior(QuackBehavior quackBehavior) {
        this.quackBehavior = quackBehavior;
    }
}

定义野鸭子:会飞,呱呱叫。

public class MallardDuck extends Duck {
    @Override
    public void display() {
        System.out.println("我是一只野鸭子");
    }
}

定义模型鸭子:不会飞也不会叫。

public class ModelDuck extends Duck {
    @Override
    public void display() {
        System.out.println("我是一只模型鸭子");
    }
}

策略抽象角色 FlyBehavior 、QuackBehavior

叫声抽象策略

public interface QuackBehavior {
    void quack();
}

飞行抽象策略

public interface FlyBehavior {
    void fly();
}

策略具体角色

实现抽象策略,规定算法逻辑。

  • 不会飞的策略
public class FlyNoWay implements FlyBehavior {
    @Override
    public void fly() {
        System.out.println("老司机带带我,我不会飞");
    }
}
  • 会飞的策略
public class FlyWithWings implements FlyBehavior {
    @Override
    public void fly() {
        System.out.println("我会飞行,一冲云霄");
    }
}
  • 叫声策略呱呱叫实现
public class Quack implements QuackBehavior {
    @Override
    public void quack() {
        System.out.println("呱呱叫");
    }
}
  • 叫声策略: 不会叫策略
public class MuteQuack implements QuackBehavior {
    @Override
    public void quack() {
        System.out.println("我不会叫");
    }
}

生成不同的鸭子

  • 会叫的野鸭子
public class MallardDuck extends Duck {
    @Override
    public void display() {
        System.out.println("我是一只野鸭子");
    }
}
  • 不会叫的模型鸭子
public class ModelDuck extends Duck {
    @Override
    public void display() {
        System.out.println("我是一只模型鸭子");
    }
}

定义显示屏展示鸭子

public class MiniDuckSimulator {
    public static void main(String[] args) {
        //定义不会叫不会飞的鸭子
        FlyBehavior flyBehavior = new FlyNoWay();
        QuackBehavior quackBehavior = new MuteQuack();
        Duck modelDuck = new ModelDuck();
        //这里我们可以设置不同的行为实现类就会执行不同的策略
        modelDuck.setFlyBehavior(flyBehavior);
        modelDuck.setQuackBehavior(quackBehavior);
        modelDuck.display();
        modelDuck.performFly();
        modelDuck.performQuack();
        System.out.println("-------------------");
        // 定义会叫会飞的鸭子
        FlyBehavior flyWithWings = new FlyWithWings();
        QuackBehavior quack = new Quack();
        Duck mallardDuck = new MallardDuck();
        mallardDuck.setFlyBehavior(flyWithWings);
        mallardDuck.setQuackBehavior(quack);
        mallardDuck.display();
        mallardDuck.performFly();
        mallardDuck.performQuack();
    }
}

其实就是将将不同的算法抽象,通过上下文切换策略实现不同行为。我们是不是还可以使用策略模式代替很多的if else 判断执行不同的算逻辑?这里留给读者去发现使用场景。


相关文章
|
17天前
|
设计模式 算法 PHP
php设计模式--策略模式(六)
php设计模式--策略模式(六)
11 0
|
3月前
|
设计模式 算法 Java
行为型设计模式-策略模式(Strategy Pattern)
行为型设计模式-策略模式(Strategy Pattern)
|
3月前
|
设计模式 算法 Java
【设计模式】策略模式在数据接收和发送场景的应用
在数据接收和发送场景打算使用了if else进行判断。ASystem.sync("向A同步数据");BSystem.sync("向B同步数据");...非常麻烦,需求多了很臃肿!策略模式(Strategy Pattern)定义了一组同类型的算法,在不同的类中封装起来,每种算法可以根据当前场景相互替换,从而使算法的变化独立于使用它们的客户端(即算法的调用者)。// 创建两个策略对象// 创建上下文对象,并传入策略对象。
59 1
|
4月前
|
设计模式 算法
设计模式思考,简单工厂模式和策略模式的区别?
设计模式思考,简单工厂模式和策略模式的区别?
|
4月前
|
设计模式 Java
细说一下设计模式中的策略模式!
细说一下设计模式中的策略模式!
30 0
|
1月前
|
设计模式 算法
策略模式--设计模式
策略模式--设计模式
17 0
|
5月前
|
设计模式 算法 关系型数据库
设计模式系列教程(13) - 策略模式
设计模式系列教程(13) - 策略模式
22 0
|
1月前
|
设计模式 算法 Java
【设计模式】策略模式
【设计模式】策略模式
|
3月前
|
设计模式 算法 自动驾驶
常见的设计模式(模板与方法,观察者模式,策略模式)
随着时间的推移,软件代码越来越庞大,随着而来的就是如何维护日趋庞大的软件系统。在面向对象开发出现之前,使用的是面向过程开发来设计大型的软件程序,面向过程开发将软件分成一个个单独的模块,模块之间使用函数进行组合,最后完成系统的开发,每次需要修改软件,如果不涉及好各个模块的关系,就会导致软件系统难以维护,从而导致软件变得不可使用。面向对象方法用对象模拟问题域中的实体,以对象间的联系刻画实体间联系
65 2
|
3月前
|
设计模式 算法
设计模式 | 策略模式
设计模式 | 策略模式
18 0