介绍
你在编程中可能会遇到过,有时一个对象的状态改变会影响很多类的行为,比较典型的就是在购买流程中,购买成功后会影响库存、销量、会通知发货、有的还需要发送短信。购买这个行为在发生变化时需要同时发生其他行为,但是如果写在一起,一旦流程发生变化,改起来会比较麻烦,将这些行为写在一起的耦合度也太高,观察者模式可以解决这个问题,通过观察者和状态主题互相注册的方式,使主体在发生变化时能够通知观察者,两者组合在一起,耦合度较低。实现一种自动更新机制。
上面这个例子好像还不能体现出观察者模式的优势,比如一个购物网站,各类商品的价格都是由汇率决定,当汇率发生改变,需要通知所有产品类目价格改变,但是产品类目的数目、有哪些都是不确定的,这让它没有办法通知,也不能让各个产品类目来主动查询,所以此时必须使用观察者模式来实现这个数据绑定,这让我想起了C#中的数据绑定机制。
实现方式
观察者模式是通过一个叫Observer的接口实现,如果某个类需要作为观察者,则需要实现这个类:
interface Observer{
function notefy($obj);
}
如果一个对象想要被观察,需要一个注册方法,这样,观察者对象可以用来注册自己,
例子
购物网站的货品价格需要根据当天汇率来进行计算。这些产品项目需要作为“观察者”观察汇率这个主体。
<?php /** * */ class ExchangeRate { static private $instance = NULL; private $observers = array(); private $exchange_rate; private function ExchangeRate(); static private function getInstance(){ if(self::$instance == NULL) self::$instance = new ExchangeRate(); return slef::$instance; } public function getExangeRate(){ return $this->$exchange_rate; } public function setExchangeRate($new_rate){ $this->exange_rate = $new_rate; $this->notifyObservers(); // 通知更新 } public function registObserver($obj){ $this->observers[] = $obj; } public notifyObservers(){ foreach ($this->observers as $obj) { $obj->notify($this); } } interface Observer{ function notify($obj); } /** * */ class ProductIten implements Observer { function __construct() { ExchangeRate::getInstance()->registObserver($this); } public function notify($obj){ if($obj instanceof ExchangeRate){ // 更新 print"数据已更新"; } } } $priduct1 = new ProductIten(); ExchangeRate::getInstance()->setExchangeRate(2); ?>