浅谈PHP设计模式的观察者模式

简介

观察者模式是行为型模式的一种,定义了对象间一对多的关系。当对象的状态发生变化时候,依赖于它的对象会得到通知。

创新互联不只是一家网站建设的网络公司;我们对营销、技术、服务都有自己独特见解,公司采取“创意+综合+营销”一体化的方式为您提供更专业的服务!我们经历的每一步也许不一定是最完美的,但每一步都有值得深思的意义。我们珍视每一份信任,关注我们的成都网站设计、成都做网站质量和服务品质,在得到用户满意的同时,也能得到同行业的专业认可,能够为行业创新发展助力。未来将继续专注于技术创新,服务升级,满足企业一站式成都营销网站建设需求,让再小的品牌网站制作也能产生价值!

适用场景

  1. 类似触发钩子事件,可做消息通知、框架底层监听。
  2. 一个对象的改变会导致一个或多个对象发生改变,方便扩展的写法。

优点

方便扩展,降低耦合,统一触发规则。当需要新增或者删除一个观察者的时候,只需要增加观察者就行。

缺点

  1. 相比于不用观察者而是直接依赖某些类,增加代码的复杂度。
  2. 如果观察者者被观察者互相依赖,有产生死循环的可能。

补充

  1. 需要理清楚观察者和被观察者是谁,观察者可以理解为被动受到通知的对象。被观察者是主动发送通知的对象。
  2. 固定的套路,被观察者至少需要一个添加观察者的方法和一个通知观察者的方法用来确定身份和发送通知(一般有三个,多一个删除观察者的方法),观察者至少需要一个更新的方法用于接收被观察者的通知。

代码(自定义实现)

//假设用户成功购买商品后需要发送邮件和短信通知
class Order {
    private $observers = [];

    //添加观察者
    public function attach($type, $observer) {
        $this->observers[$type] = $observer;
    }

    //对每个观察者进行通知
    public function notify() {
        if ($this->observers == []) {
            return null;
        }

        foreach ($this->observers as $every_observer) {
            (new $every_observer)->update($this);
        }
    }


    //购买商品,触发通知
    public function buyGoods() {
        //todo 订单操作
        echo '商品购买完成' . PHP_EOL;
        $this->notify();
    }
}

class Mail {
    public function update($observer) {
        echo '发送电子邮件' . PHP_EOL;
    }
}

class Sms {
    public function update($observer) {
        echo '发送短信' . PHP_EOL;
    }
}


$order = new Order();
//添加观察者
$order->attach('mail', Mail::class);
$order->attach('sms', Sms::class);

$order->buyGoods();

代码(基于SPL实现)

SPL(Standard PHP Library)标准PHP类库,用于解决典型问题的一组接口与类的集合。

class OrderListener implements \SplSubject {
    //观察者列表
    public $observers;

    public function __construct() {
        //SplObjectStorage类提供从对象到数据的映射,或者通过忽略数据,提供对象集的映射。在许多需要唯一标识对象的情况下,这种双重用途非常有用。
        $this->observers = new \SplObjectStorage();
    }


    //添加要通知的对象
    public function attach(\SplObserver $observer) {
        $this->observers->attach($observer);
    }

    //移除要通知的对象
    public function detach(\SplObserver $observer) {
        $this->observers->detach($observer);
    }


    //通知
    public function notify() {
        //将迭代器(此处可以理解为指针)倒回到第一个存储元素。
        $this->observers->rewind();
        //判断指针是否有效
        while($this->observers->valid()) {
            //获取当前的观察者
            $curr_obj = $this->observers->current();
            //对当前观察者进行通知
            $curr_obj->update($this);
            //向下移动指针
            $this->observers->next();
        }
    }


    //触发通知
    public function buyGoods() {
        echo '购买成功' . PHP_EOL;
        $this->notify();
    }
}

//SplObserver接口与SplSubject接口一起使用,以实现观察者设计模式。
class Mail implements \SplObserver {
    //对被观察的对象做相应的处理
    public function update(\SplSubject $subject) {
        echo '发送邮件' . PHP_EOL;
    }
}

class Sms implements \SplObserver {
    //对被观察的对象做相应的处理
    public function update(\SplSubject $subject) {
        echo '发送短信' . PHP_EOL;
    }
}


$listener = new OrderListener();
//添加观察者
$listener->attach(new Mail());
$listener->attach(new Sms());

$listener->buyGoods();

通知代码(基于SPL实现的notify方法优化)

	//以上代码的notify方法使用原生手动调整指针的方式去实现。也可以使用foreach去遍历实现
    public function notify() {
        foreach ($this->observers as $observer) {
            $observer->update($this);
        }
    }

网页题目:浅谈PHP设计模式的观察者模式
文章地址:http://csdahua.cn/article/dsoihdj.html
扫二维码与项目经理沟通

我们在微信上24小时期待你的声音

解答本文疑问/技术咨询/运营咨询/技术建议/互联网交流