PHP觀察者模式
1、什么是觀察者模式?
從面向過程的角度來看,首先是觀察者向主題注冊,注冊完之后,主題再通知觀察者做出相應的操作,整個事情就完了
從面向對象的角度來看,主題提供注冊和通知的接口,觀察者提供自身操作的接口。(這些觀察者擁有一個同一個接口。)觀察者利用主題的接口向主題注冊,而主題利用觀察者接口通知觀察者。耦合度相當之低
流程圖如下:
2、為什么要用觀察者模式?
觀察者模式更多體現了兩個獨立的類利用接口完成一件本應該很復雜的事情。不利用主題類的話,我們還需要不斷循環創建實例,執行操作。而現在只需要創建實例就好,執行操作的事兒只需要調用一次通知的方法就好啦
3、應用場景
當一個對象的改變需要同時改變其他對象的時候,而且它不知道具體有多少對象有待改變時,應該考慮使用觀察者模式。
4、示例代碼
/** * @purpose: 觀察者接口, 定義觀察者具體需要執行的方法,當然方法名和方法個數可以自定義 * Interface Observer */ interface Observer { /** * @purpose: 廣播通知后,所有已注冊的觀察者都需要執行該方法。 * @return mixed */ public function eat(); } /** * @purpse: 定義貓貓類,繼承觀察者接口,實現具體細節 * Class Cat */ class Cat implements Observer{ public function eat(){ echo 'Cat eat fish'; } } /** * @purpse: 定義狗狗類,繼承觀察者接口,實現具體細節 * Class Dog */ class Dog implements Observer{ public function eat(){ echo 'Dog eat bones'; } } /** * @purpose: 主題接口, 定義添加觀察者和廣播通知的方法 * Interface Notify */ interface Subject { /** * @purpose: 添加觀察者 * @param string $key 給所添加的觀察者的一個唯一 key,方便從注冊樹中移除觀察者 * @param Observer $observer 觀察者對象 * @return mixed */ public function addObserver($key, Observer $observer); /** * @purpose: 從注冊樹中移除觀察者 * @param string $key 給所添加的觀察者的一個唯一 key,方便從注冊樹中移除觀察者 * @return mixed */ public function removeObserver($key); /** * @purpose: 廣播通知以注冊的觀察者 * @return mixed */ public function notify(); } /** * @purpose: 實現主體接口,主要就是添加觀察者和廣播通知觀察者 * Class Action */ class Action implements Subject { /** * @var array 保存所有已注冊的觀察者 */ public $_observer = []; /** * @purpose: 添加觀察者 * @param string $key 給所添加的觀察者的一個唯一 key,方便從注冊樹中移除觀察者 * @param Observer $observer 觀察者對象 * @return mixed */ public function addObserver($key, Observer $observer) { $this->_observer[$key] = $observer; } /** * @purpose: 從注冊樹中移除觀察者 * @param string $key 給所添加的觀察者的一個唯一 key,方便從注冊樹中移除觀察者 * @return mixed */ public function removeObserver($key) { unset($this->_observer[$key]); } /** * @purpose: 廣播通知以注冊的觀察者,對注冊樹進行遍歷,讓每個對象實現其接口提供的操作 * @return mixed */ public function notify() { foreach ($this->_observer as $observer) { $observer->eat(); } } }