觀察者模式(有時又被稱為發布/訂閱模式)是軟體設計模式的一種。
在此種模式中,一個目標對象管理所有相依於它的觀察者對象,並且在它本身的狀態改變時主動發出通知。
這通常透過呼叫各觀察者所提供的方法來實現。
實現觀察者模式的時候要注意,觀察者和被觀察對象之間的互動關係不能
體現成類之間的直接調用,否則就將使觀察者和被觀察對象之間緊密的耦合起來,
從根本上違反物件導向的設計的原則。無論是觀察者“觀察”觀察對象,
還是被觀察者將自己的改變“通知”觀察者,都不應該直接調用。
通俗點說就是A對象(被觀察)通知另一個(一些)對象(觀察者)自己發生改變了,改變了什麼,至於你們這些對象要做什麼就不關我的事了,你們自己做去吧!耦合度就此降低了。。。
下面的例子使用ruby的module實現了較為傳統的觀察者模式。使用module的好處是:subject類可能是其他基類的子類,mixin實現了類似多繼承的效果。
module Subject def initialize @observers = [] end def add_observer ob @observers << ob end def delete_observer ob @observers.delete ob end def notify_observers @observers.each do |ob| ob.update self end endendclass Employee include Subject attr_reader :name, :title attr_reader :salary def initialize name, title, salary super() @name = name @title = title @salary = salary end def salary=new_salary @salary = new_salary notify_observers endendclass Taxman def update obj puts "#{obj.name} now has a salary of #{obj.salary}" endendjack = Employee.new('jack', 'prgramer', 3000)jack.add_observer(Taxman.new)jack.salary = 3000
我們可以自己實現Subject module,不過這樣做有些畫蛇添足,因為ruby核心庫本身就包含了Observable模組,我們只需要將其mixin代碼既可。
require 'observer'class Employee include Observable attr_reader :name, :title, :salary def initialize name, title, salary @name = name @title = title @salary = salary end def salary=(new_salary) @salary = new_salary changed notify_observers(self) end # salary=end # Employee
在notify_observers之前必須先調用changed方法,表示確實發生了改變,否則notify_observers方法是無效的。