正文

一、定义

策略模式定义了算法族,分别封装起来,让它们之间可以相互替换,此模式让算法的变化独立于使用算法的客户。

要点:

  • 策略模式把系统中会变化的部分抽出来封装。

二、实现步骤

1、创建策略接口

/**
* 策略接口
*/
public interface Strategy { /**
* 执行策略行为
*/
public void perform();
}

2、创建策略接口的实现类

(1)策略实现类 A

/**
* 策略实现类A
*/
public class StrategyImplA implements Strategy { /**
* A策略行为
*/
@Override
public void perform() {
System.out.println("perform A...");
}
}

(2)策略实现类 B

/**
* 策略实现类B
*/
public class StrategyImplB implements Strategy { /**
* B策略行为
*/
@Override
public void perform() {
System.out.println("perform B...");
}
}

3、在使用策略的类中,声明并使用接口类型的策略变量

/**
* 策略使用者
*/
public class StrategyUser { /**
* 声明接口类型的策略变量
*/
private Strategy strategy; /**
* 通过构造实例化策略
*/
public StrategyUser(Strategy strategy) {
this.strategy = strategy;
} /**
* 执行策略使用者的行为
*/
public void doBehavior() {
// do something... // 使用策略
strategy.perform(); // do something...
}
}

4、通过实例化不同的策略实现类,来改变使用者的行为

public class Test {

    public static void main(String[] args) {
// 使用策略A
StrategyUser userA = new StrategyUser(new StrategyImplA());
userA.doBehavior();
// 使用策略B
StrategyUser userB = new StrategyUser(new StrategyImplB());
userB.doBehavior();
}
}

三、举个栗子

1、背景

Joe 上班的公司做了一套相当成功的模拟鸭子游戏:SimUDuck。游戏中会出现各种鸭子,鸭子的种类及属性如下:

  • 种类:绿头鸭、红头鸭、橡皮鸭、诱饵鸭。
  • 属性:外观、游泳行为、飞行行为、呱呱叫行为(叫声)。

不同种类的鸭子所对应的属性如下:

  • 绿头鸭:绿头鸭的外观、会游泳、会飞行、呱呱叫。
  • 红头鸭:红头鸭的外观、会游泳、会飞行、呱呱叫。
  • 橡皮鸭:橡皮鸭的外观、会游泳(漂浮)、不会飞行、吱吱叫。
  • 诱饵鸭:诱饵鸭的外观、会游泳(漂浮)、不会飞行、不会叫。

2、要点

  • 由于不同种类的鸭子可能具有不同的飞行行为、呱呱叫行为,因此,可以使用策略模式把这两种行为抽出来。

3、实现

(1)创建行为接口

/**
* 飞行行为接口
*/
public interface FlyBehavior { public void fly();
}
/**
* 呱呱叫行为接口
*/
public interface QuackBehavior { public void quark();
}

(2)实现行为接口

/**
* 用翅膀飞行
*/
public class FlyWithWings implements FlyBehavior { @Override
public void fly() {
System.out.println("I'm flying!");
}
} /**
* 不会飞行
*/
public class FlyNoWay implements FlyBehavior { @Override
public void fly() {
System.out.println("I can't flying!");
}
}
/**
* 呱呱叫
*/
public class Quack implements QuackBehavior { @Override
public void quark() {
System.out.println("Quack");
}
} /**
* 吱吱叫
*/
public class Squeak implements QuackBehavior { @Override
public void quark() {
System.out.println("Squeak");
}
} /**
* 不会叫
*/
public class MuteQuack implements QuackBehavior { @Override
public void quark() {
System.out.println("<<Silence>>");
}
}

(3)创建鸭子抽象类,并使用行为接口

/**
* 鸭子抽象类
*/
public abstract class Duck { FlyBehavior flyBehavior;
QuackBehavior quackBehavior; public Duck() {
} /**
* 外观
*/
public abstract void display(); /**
* 游泳行为
*/
public void swim() {
System.out.println("All ducks float, even decoys!");
} /**
* 飞行行为
*/
public void performFly() {
flyBehavior.fly();
} /**
* 呱呱叫行为
*/
public void performQuark() {
quackBehavior.quark();
}
}

(4)创建鸭子子类,并指定具体的行为实现

/**
* 绿头鸭
*/
public class MallardDuck extends Duck { public MallardDuck() {
// 指定具体的飞行行为
flyBehavior = new FlyWithWings();
// 指定具体的呱呱叫行为
quackBehavior = new Quack();
} @Override
public void display() {
System.out.println("I'm a real Mallard duck");
}
} /**
* 红头鸭
*/
public class RedHeadDuck extends Duck { public RedHeadDuck() {
// 指定具体的飞行行为
flyBehavior = new FlyWithWings();
// 指定具体的呱呱叫行为
quackBehavior = new Quack();
} @Override
public void display() {
System.out.println("I'm a red head duck");
}
} /**
* 橡皮鸭
*/
public class RubberDuck extends Duck { public RubberDuck() {
// 指定具体的飞行行为
flyBehavior = new FlyNoWay();
// 指定具体的呱呱叫行为
quackBehavior = new Squeak();
} @Override
public void display() {
System.out.println("I'm a rubber duck");
}
} /**
* 诱饵鸭
*/
public class DecoyDuck extends Duck { public DecoyDuck() {
// 指定具体的飞行行为
flyBehavior = new FlyNoWay();
// 指定具体的呱呱叫行为
quackBehavior = new MuteQuack();
} @Override
public void display() {
System.out.println("I'm a decoy duck");
}
}

(5)测试

public class Test {

    public static void main(String[] args) {
// 绿头鸭
MallardDuck mallardDuck = new MallardDuck();
mallardDuck.performFly();
mallardDuck.performQuark();
// 红头鸭
RedHeadDuck redHeadDuck = new RedHeadDuck();
redHeadDuck.performFly();
redHeadDuck.performQuark();
// 橡皮鸭
RubberDuck rubberDuck = new RubberDuck();
rubberDuck.performFly();
rubberDuck.performQuark();
// 诱饵鸭
DecoyDuck decoyDuck = new DecoyDuck();
decoyDuck.performFly();
decoyDuck.performQuark();
}
}

《Head First 设计模式》:策略模式的更多相关文章

  1. 《Android源码设计模式》学习笔记之ImageLoader

    微信公众号:CodingAndroid cnblog:http://www.cnblogs.com/angel88/ CSDN:http://blog.csdn.net/xinpengfei521 需 ...

  2. 《Android源码设计模式》--抽象工厂模式

    No1: 4种MediaPlayer Factory分别会生成不同的MediaPlayer基类:StagefrightPlayer.NuPlayerDriver.MidiFile和TestPlayer ...

  3. 《Android源码设计模式》--Builder模式

    No1: 将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示 No2: 在Android源码中,最常用到的Builder模式就是AlertDialog.Builder No3: ...

  4. 《Android源码设计模式》--策略模式

    No1: 定义:策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换.策略模式让算法独立于使用它的客户而独立变化. No2: 使用场景: 1)针对同一类型问题的多种处理方式,仅 ...

  5. 《Android源码设计模式》--模板方法模式

    No1: 模板方法模式包括:抽象类(其中定义了一系列顺序方法).具体实现类A.具体实现类B 如果子类有实现不一样的细节,重写父类的某个方法即可 No2: AsyncTask对象调用execute方法后 ...

  6. 《Android源码设计模式》--状态模式--责任链模式--解释器模式--命令模式--观察者模式--备忘录模式--迭代器模式

    [状态模式] No1: Wifi设置界面是一个叫做WifiSetting的Fragment实现的 No2: 在不同的状态下对于扫描Wifi这个请求的处理是完全不一样的.在初始状态下扫描请求被直接忽略, ...

  7. 《Android源码设计模式》--享元模式

    No1: 享元模式是对象池的一种实现.享元模式用来尽可能减少内存使用量,它适合用于可能存在大量重复对象的场景,来缓存可共享的对象,达到对象共享.避免创建过多对象的效果,这样一来就可以提升性能.避免内存 ...

  8. 《Android源码设计模式》--工厂方法模式

    No1: 对于一个应用程序来说,其真正的入口是在ActivityThread类中,ActivityThread中含有我们熟悉的main方法.ActivityThread是一个final类,不能被继承. ...

  9. 《Android源码设计模式》--原型模式

    No1: 原型模式使用场景: 1)类初始化需要消耗非常多的资源,这个资源包括数据.硬件资源等,通过原型复制避免这些消耗 2)通过new产生一个对象需要非常繁琐的数据准备货访问权限,这是可以使用原型模式 ...

  10. 《Android源码设计模式》--装饰模式

    No1: Activity继承于ContextThemeWrapper,继承于ContextWrapper,继承于Context. No2: Context中方法的所有实现均由ContextImpl类 ...

随机推荐

  1. POJ - 3074 Sudoku (搜索)剪枝+位运算优化

    In the game of Sudoku, you are given a large 9 × 9 grid divided into smaller 3 × 3 subgrids. For exa ...

  2. SpringBoot + MybatisPlus3.x 代码生成

    主要参考另外两篇博文,这里表示感谢 参考一(mybatisplus3.x分页) : https://www.jianshu.com/p/2ec9337dc2b0 参考二(mybatisplus2.x升 ...

  3. ubuntu 和 centos 如何区分系统

    Ubuntu Ubuntu有着漂亮的用户界面,完善的包管理系统,强大的软件源支持,丰富的技术社区,Ubuntu还对大多数硬件有着良好的兼容性,包括最新的图形显卡等等.这一切让Ubuntu越来越向大众化 ...

  4. 入职半年多,Guide哥总结了这些帮助你领先90%程序员的好习惯!

    大家好,我是 Guide 哥,这篇文章还是不分享技术文章了,我想简单聊聊一些程序员的好习惯. 其实自己写这篇文章还是很忐忑的,毕竟自己算不上一个优秀的程序员,但也不是那么差.大学学习编程 4 年,工作 ...

  5. 201771010113 李婷华 《面向对象程序设计(java)》

    一.理论知识部分 设计模式(Design pattern)是设计者一种流行的思考设计问题的方法,是一套被反复使用,多数人知晓的,经过分类编目的,代码设计经验的总结.使用设计模式是为了可重用代码.让代码 ...

  6. .Net Core WPF之XAML概述

    原文链接,机器翻译,有误处参看原文. XAML overview in WPF 2019/08/08 What is XAML XAML syntax in brief Case and white ...

  7. QtCreator中使用链接库

    说明 之前讨论的DLL的静态链接和动态连接都是基于 MSVC 编译器,但是 MinGW 似乎有另外一套类似但是不相同的机制.下文均在 windows 下使用 Qt Creator 中使用 MinGW ...

  8. 【Spark】快来学习RDD的创建以及操作方式吧!

    目录 RDD的创建 三种方式 从一个集合中创建 从文件中创建 从其他的RDD转化而来 RDD编程常用API 算子分类 Transformation 概述 帮助文档 常用Transformation表 ...

  9. [hdu4609]计数方法,FFT

    题目:给一个数组a,从里面任选三个数,求以这三个数为三条边能构成三角形的概率. 思路:由于每个数只能用一次,所以考虑枚举三边中的最大边.先将a数组排序,然后枚举它的每个数x作为最大边,那么问题就是要求 ...

  10. 800+Java后端经典面试题,希望你找到自己理想的Offer呀~

    前言 在茫茫的互联网海洋中寻寻觅觅,我收藏了800+道Java经典面试题,分享给你们.建议大家收藏起来,在茶余饭后拿出来读一读,以备未雨绸缪之需.另外,面试题答案的话,我打算后面慢慢完善在github ...