设计模式之策略模式

思考:

  • 假设有个需求,模拟鸭子游戏:在游戏中会出现各种各样的鸭子,一边游泳戏水,一边呱呱叫。
    开始我们的设计吧:
 class Duck
{

    public function quack()
    {
        //鸭子呱呱叫
    }

    public function fly()
    {
        //鸭子会飞
    }

    public function display()
    {
        //鸭子外观
    }
    //...鸭子的其他方法
}
class GreenDuck extends Duck
{
    public function display()
    {
        //外观是绿色的
    }
}
class RedDuck extends Duck
{
    public function display()
    {
        //外观是红色的
    }
}

这应该是大家最容易想到的设计吧,没学习设计模式之前,我也只能想到这样的设计。这样设计有什么坏处了:
1.所有继承了Duck超类的子类都拥有了Duck超类中的方法,实际上并不是所有子类都需要父类的该方法。除非子类覆盖超类的方法。

public function quack() {
    //覆盖,然后什么都不做
}

2.如果要让所有鸭子的叫声各不相同,那么必须在子类中重写父类的方法,假设有5个子类,其中2个子类都重写了父类fly()方法,且代码一样:

在GreenDuck类中:
public function fly()
{
    echo '我可以垂直起飞';
}
在RedDuck类中:
public function fly()
{
    echo '我可以垂直起飞';
}

此时两个类中的fly()方法代码完全一样,造成了代码重复,且后期一旦修改的话就必须依次找到这几个类,然后一起修改代码。如果需求变化很大的话,那维护起来真的不敢想象。

现在引出一个设计原则:
找出应用中可能需要变化之处,把它们独立出来,不要和那些不需要变化的代码混在一起。

我们发现鸭子的fly()、quack()方法会根据鸭子类的不同而不断变化,那么我们根据该原则,我们应当把飞行的行为和叫的行为从Duck类中取出独立起来。

此时,引出另外一个设计原则:
针对接口编程,而不是针对实现编程。

官方描述感觉太抽象,还是讲人话吧:

子类的鸭子继承了超类Duck类,其行为是子类自行重写方法实现的或继承自父类的方法而来的。这两种做法都是依赖于"实现"。我们被实现绑的死死的,除非写更多的代码,否则没办法更改行为。
而针对接口编程,意思是针对超类型编程,变量的声明应该是超类,超类一般是一个抽象方法或接口。
请看例子:

针对实现编程

$duck = new GreenDuck();
$duck->fly();
子类直接调用父类的方法。

--------------------------------------------
针对接口编程:
飞行的行为方法来自另外一个类(飞行类),构造函数中引入这个类并赋值给一个属性。  
注意,这时实现飞行行为则是飞行类调用自己的fly()的方法,而不是调用父类的方法。
abstract class Duck
{
    protected $flyBehavior;
    /**
     * Duck constructor.
     * @param $flyBehavior
     */
    public function __construct(FlyBehavior $flyBehavior)
    {
        $this->flyBehavior = $flyBehavior;
        $this->quackBehavior = $quackBehavior;
    }


    /**
     * @param mixed $flyBehavior
     */
    public function setFlyBehavior($flyBehavior)
    {
        $this->flyBehavior = $flyBehavior;
    }
    public function performFly()
    {
        $this->flyBehavior->fly();
    }

    /**
     * 展示
     */
    abstract public function display();
}

好了,现在上全部代码:
Duck.php

<?php
namespace App\StrategyPattern;
abstract class Duck
{
    protected $flyBehavior;
    protected $quackBehavior;

    /**
     * Duck constructor.
     * @param $flyBehavior
     * @param $quackBehavior
     */
    public function __construct(FlyBehavior $flyBehavior, QuackBehavior $quackBehavior)
    {
        $this->flyBehavior = $flyBehavior;
        $this->quackBehavior = $quackBehavior;
    }


    /**
     * @param mixed $flyBehavior
     */
    public function setFlyBehavior($flyBehavior)
    {
        $this->flyBehavior = $flyBehavior;
    }

    /**
     * @param mixed $quackBehavior
     */
    public function setQuackBehavior($quackBehavior)
    {
        $this->quackBehavior = $quackBehavior;
    }

    public function performFly()
    {
        $this->flyBehavior->fly();
    }

    public function performQuack()
    {
        $this->quackBehavior->quack();
    }

    public function swim()
    {
        echo '我是一只鸭子,所以我肯定是会游泳的,没有例外哦!';
    }

    /**
     * 展示
     */
    abstract public function display();

}

FlyBehavior.php

<?php
namespace App\StrategyPattern;
interface FlyBehavior
{
    public function fly();
}

FlyNoWay.php

<?php
namespace App\StrategyPattern;
class FlyNoWay implements FlyBehavior
{
    public function fly()
    {
        echo '我是不会飞的鸭子哦';
    }
}

FlyWithWings.php

<?php
namespace App\StrategyPattern;
class FlyWithWings implements FlyBehavior
{
    public function fly()
    {
        echo '我是会飞的鸭子哦';
    }
}

QuackBehavior.php

<?php
namespace App\StrategyPattern;
interface QuackBehavior
{
    public function quack();
}

Quack.php

<?php
namespace App\StrategyPattern;
class Quack implements QuackBehavior
{
    public function quack()
    {
        echo '呱呱叫';
    }
}

Squeak.php

<?php
namespace App\StrategyPattern;
class Squeak implements QuackBehavior
{
    public function quack()
    {
        echo '吱吱叫';
    }
}

RedDuck.php

<?php
namespace App\StrategyPattern;
class RedDuck extends Duck
{
    public function display()
    {
        echo '我是一只红头呀,哇哈哈';
    }
}

具体调用示例如下:

<?php
use App\StrategyPattern\RedDuck;
use App\StrategyPattern\FlyWithWings;
use App\StrategyPattern\Squeak;
$duck = new RedDuck(new FlyWithWings(),new Squeak());
$duck->performQuack();
$duck->performFly();
$duck->display();

终端调用输出:

☁  learn_design_pattern  php strategy.php
吱吱叫我是会飞的鸭子哦我是一只红头呀,哇哈哈

现在给出策略模式的官方定义:

策略模式

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

算法族,即是刚才各自定义的行为接口(FlyBehavior.php和QuackBehavior.php)。使用算法的客户即是Duck类。
请大家再次理解 针对接口编程 的真正意义,是在客户中定义了算法族行为属性,然后通过属性调用算法族的方法。
声明:

设计模式学习我选用的教材是《Head First 设计模式》,此篇讲解代码也是出自该书。写这篇文章的目的也是为了自己做个笔记,同时可以和大家一起交流。在写作的过程中,我发现设计模式要讲透彻真的非常困难,里面包含了太多的设计思想。所以后续我尽量做到自己理解的非常非常透彻以后才会发到简书上和大家一起交流。
以上如有错误的地方请及时指出,谢谢。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容

  • 客户需求 程序设计 1、直接利用继承如何? 将以上四种行为全部写到Duck这个基类中,然后子类重写飞和叫的行为。但...
    BlainPeng阅读 466评论 2 7
  • 策略模式,是我们接触到的第一个设计模式,也是较容易理解的一个模式。我们可以给它下一个定义:** 定义了算法族,分别...
    六尺帐篷阅读 700评论 0 8
  • 一直想把常见的设计模式系统地学习一遍,结果和大多数人一样,过了几天就没能坚持下去了。我发现学习这件事情急不得,往往...
    Neulana阅读 561评论 5 2
  • 模拟鸭子 模拟一个鸭子的游戏SimUDuck,游戏中会出现各种鸭子,一边游戏戏水,一边呱呱叫。现采用oo技术,设计...
    yaSecrets阅读 247评论 0 0
  • 使用模式最好的方式是:“把模式装进脑子里,然后在你的设计和已有的应用中,寻找何处可以使用他们。”,以往是代码复用,...
    秒赞不是偶然阅读 275评论 0 0