作者:Grey

原文地址:http://www.cnblogs.com/greyzeng/p/5915202.html

模式名称

策略模式(Strategy Pattern)

需求

模拟鸭子游戏,游戏中会出现各种鸭子,一边游泳戏水,一边呱呱叫。我们应该如何设计这个鸭子类呢?

解决方案

首先,我们可以考虑设计一个鸭子的超类Duck,抽象出鸭子的公共方法,然后其他鸭子类型继承这个超类,对公共方法有自己独有的实现。

public abstract class Duck {

    public abstract void display();
public abstract void quack(); public void swim() {
System.out.println("all ducks can swim");
} }
public class RedHeadDuck extends Duck {

    @Override
public void display() {
System.out.println("This is RedHeadDuck");
} @Override
public void quack() {
System.out.println("RedHeadDuck Quack");
}
}
public class ModelDuck extends Duck {
@Override
public void display() {
System.out.println("This is ModelDuck");
} @Override
public void quack() {
System.out.println("ModelDuck Can not Quack");
}
}

但是,这种设计,会出现一个问题:

如果要增加一个可以飞的鸭子,我们在超类中增加一个fly()方法,所有继承这个超类的子类都会有这个方法,导致代码在多个子类中重复。

public abstract class Duck {

    public abstract void display();
public abstract void quack();
public abstract void fly(); public void swim() {
System.out.println("all ducks can swim");
} }
public class RedHeadDuck extends Duck{

    @Override
public void display() {
System.out.println("This is RedHeadDuck");
} @Override
public void quack() {
System.out.println("Gua Gua Gua~~");
} @Override
public void fly() {
System.out.println("fly fly fly~~");
}
}
public class ModelDuck extends Duck{
@Override
public void display() {
System.out.println("This is ModelDuck");
} @Override
public void quack() {
System.out.println("Silence");
} @Override
public void fly() {
// do nothing
}
}

更好的一种方法:

分开变化与不变化的部分,因为Duck类里面fly()quack()会随着鸭子的不同而改变,所以我们需要把这两个行为从Duck里面分离出来,建立一组新类来代表每个行为

QuackBehavior接口:

interface QuackBehavior {
void quack();
}

QuackBehavior 具体行为:

public class MuteQuack implements QuackBehavior {
public void quack() {
System.out.println("Silence");
}
}
public class Gua implements QuackBehavior {

    public void quack() {
System.out.println("Gua Gua Gua~~");
}
}

FlyBehavior接口:

interface FlyBehavior {
void fly();
}

FlyBehavior具体行为:

public class FlyWithWings implements FlyBehavior {
public void fly() {
System.out.println("fly fly fly~~~");
}
}
public class FlyNoWay implements FlyBehavior {
public void fly() {
// do nothing
}
}

现在鸭子会将飞行和呱呱叫的动作“委托”给FlyBehaviorQuackBehavior处理,我们在Duck中增加两个方performFly(),performQuack()

public abstract class Duck {
Duck() {} public FlyBehavior flyBehavior;
public QuackBehavior quackBehavior; public abstract void display(); public void swim() {
System.out.println("all ducks can swim");
}
public void performFly() {
flyBehavior.fly();
}
public void performQuack() {
quackBehavior.quack();
}
}

在每种鸭子继承超类Duck的时候,只需要在构造函数中设置对应的FlyBehaviorQuackBehavior实例变量即可。

public class RedHeadDuck extends Duck {
RedHeadDuck() {
this.quackBehavior = new Gua();
this.flyBehavior = new FlyWithWings();
} @Override
public void display() {
System.out.println("This is RedHeadDuck");
}
public void performFly() {
flyBehavior.fly();
}
public void performQuack() {
quackBehavior.quack();
} }
public class ModelDuck extends Duck {
ModelDuck() {
this.flyBehavior = new FlyNoWay();
this.quackBehavior = new MuteQuack();
}
@Override
public void display() {
System.out.println("This is ModelDuck");
}
public void performFly() {
flyBehavior.fly();
}
public void performQuack() {
quackBehavior.quack();
} }

同时,我们可以在Duck类中增加对FlyBehaviorQuackBehaviorsetter/getter方法,这样我们就可以动态改变鸭子的行为了

public abstract class Duck {
Duck() {} public FlyBehavior flyBehavior;
public QuackBehavior quackBehavior; public abstract void display(); public void swim() {
System.out.println("all ducks can swim");
}
public void performFly() {
flyBehavior.fly();
}
public void performQuack() {
quackBehavior.quack();
} public void setFlyBehavior(FlyBehavior flyBehavior) {
this.flyBehavior = flyBehavior;
} public void setQuackBehavior(QuackBehavior quackBehavior) {
this.quackBehavior = quackBehavior;
}
}

测试代码:

public class DuckTest {
public static void main(String[] args) {
Duck redHeadDuck = new RedHeadDuck();
redHeadDuck.display();
redHeadDuck.performFly();
redHeadDuck.performQuack(); System.out.println(); Duck modelDuck = new ModelDuck();
modelDuck.display();
modelDuck.performFly();
modelDuck.performQuack(); System.out.println(); System.out.println("I want that modelduck can fly");
modelDuck.setFlyBehavior(new FlyWithWings());
modelDuck.performFly();
}
}

输出结果:

This is RedHeadDuck
fly fly fly~~~
Gua Gua Gua~~ This is ModelDuck
can not fly
Silence I want that modelduck can fly
fly fly fly~~~

完整代码: Strategy Pattern Source

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

  1. 设计模式:策略模式(Strategy)

    定   义:它定义了算法家族,分别封装起来,让它们之间可以互相替换,此模式让算法的变化, 不会影响到使用算法的客户. 示例:商场收银系统,实现正常收费.满300返100.打8折.......等不同收费 ...

  2. PHP设计模式之策略模式

    前提: 在软件开发中也常常遇到类似的情况,实现某一个功能有多种算法或者策略,我们可以根据环境或者条件的不同选择不同的算法或者策略来完成该功能.如查 找.排序等,一种常用的方法是硬编码(Hard Cod ...

  3. JavaScript设计模式之策略模式(学习笔记)

    在网上搜索“为什么MVC不是一种设计模式呢?”其中有解答:MVC其实是三个经典设计模式的演变:观察者模式(Observer).策略模式(Strategy).组合模式(Composite).所以我今天选 ...

  4. 乐在其中设计模式(C#) - 策略模式(Strategy Pattern)

    原文:乐在其中设计模式(C#) - 策略模式(Strategy Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 策略模式(Strategy Pattern) 作者:webabc ...

  5. JavaScript设计模式之策略模式

    所谓"条条道路通罗马",在现实中,为达到某种目的往往不是只有一种方法.比如挣钱养家:可以做点小生意,可以打分工,甚至还可以是偷.抢.赌等等各种手段.在程序语言设计中,也会遇到这种类 ...

  6. 【设计模式】【应用】使用模板方法设计模式、策略模式 处理DAO中的增删改查

    原文:使用模板方法设计模式.策略模式 处理DAO中的增删改查 关于模板模式和策略模式参考前面的文章. 分析 在dao中,我们经常要做增删改查操作,如果每个对每个业务对象的操作都写一遍,代码量非常庞大. ...

  7. [design-patterns]设计模式之一策略模式

    设计模式 从今天开始开启设计模式专栏,我会系统的分析和总结每一个设计模式以及应用场景.那么首先,什么是设计模式呢,作为一个软件开发人员,程序人人都会写,但是写出一款逻辑清晰,扩展性强,可维护的程序就不 ...

  8. 设计模式入门,策略模式,c++代码实现

    // test01.cpp : Defines the entry point for the console application.////第一章,设计模式入门,策略模式#include &quo ...

  9. 设计模式之策略模式和状态模式(strategy pattern & state pattern)

    本文来讲解一下两个结构比较相似的行为设计模式:策略模式和状态模式.两者单独的理解和学习都是比较直观简单的,但是实际使用的时候却并不好实践,算是易学难用的设计模式吧.这也是把两者放在一起介绍的原因,经过 ...

  10. python设计模式之策略模式

    每次看到项目中存在大量的if else代码时,都会心生一丝不安全感. 特别是产品给的需求需要添加或者更改一种if条件时,生怕会因为自己的疏忽而使代码天崩地裂,哈哈,本文的目的就是来解决这种不安全感的, ...

随机推荐

  1. Xcode模拟器启动不了,修复ios模拟器

    1.如果可以重置模拟器 首先试试重置模拟器 2.如果不能重置,可以选择使用如下命令杀死模拟器服务: killall -9 com.apple.CoreSimulator.CoreSimulatorSe ...

  2. SQLServer事务同步下如何收缩日志

    事务同步是SQLServer做读写分离的一种常用的方式. 随着业务数据的不断增长,数据库积攒了大量的日志,为了腾出硬盘空间,需要对数据库日志进行清理 订阅数据库的日志清理 因为订阅数据库所有的数据都来 ...

  3. Android混合开发之WebView与Javascript交互

    前言: 最近公司的App为了加快开发效率选择了一部分功能采用H5开发,从目前市面的大部分App来讲,大致分成Native App.Web App.Hybrid App三种方式,个人觉得目前以Hybri ...

  4. .NET Core的文件系统[5]:扩展文件系统构建一个简易版“云盘”

    FileProvider构建了一个抽象文件系统,作为它的两个具体实现,PhysicalFileProvider和EmbeddedFileProvider则分别为我们构建了一个物理文件系统和程序集内嵌文 ...

  5. EditText 基本用法

    title: EditText 基本用法 tags: EditText,编辑框,输入框 --- EditText介绍: EditText 在开发中也是经常用到的控件,也是一个比较必要的组件,可以说它是 ...

  6. ASP.NET Core project.json imports 是什么意思?

    示例代码: "frameworks": { "netcoreapp1.0.0": { "imports" : "portable- ...

  7. javascript动画系列第二篇——磁性吸附

    × 目录 [1]范围限定 [2]拖拽范围 [3]磁性吸附 前面的话 上一篇,我们介绍了元素拖拽的实现.但在实际应用中,常常需要为拖拽的元素限定范围.而通过限定范围,再增加一些辅助的措施,就可以实现磁性 ...

  8. FullCalendar应用——整合农历节气和节日

    FullCalendar用来做日程管理功能非常强大,但是唯一不足的地方是没有将中国农历历法加进去,今天我将结合实例和大家分享如何将中国农历中的节气和节日整合到FullCalendar中,从而增强其实用 ...

  9. Atitit 软件工程概览attilax总结

    Atitit 软件工程概览attilax总结 1.1. .2 软件工程的发展 进一步地,结合人类发展史和计算机世界演化史来考察软件工程的发展史. 表2 软件工程过程模型 表2将软件工程的主要过程模型做 ...

  10. angularJS(6)

    angularJS(6) 一:angularJs的事件. 1.ng-click指令定义了AngularJS点击事件. <div ng-app="myapp" ng-contr ...