軟件設(shè)計(jì)中會碰到這樣的關(guān)系:一個對象依賴于另一個對象,必須根據(jù)后者的狀態(tài)更新自己的狀態(tài),可以把后者稱作目標(biāo)對象,前者稱作觀察者對象。不但觀察者依賴于目標(biāo),當(dāng)目標(biāo)的狀態(tài)改變時也要通知觀察者,這就出現(xiàn)了雙向的依賴。兩個對象互相依賴的后果是它們必須一起復(fù)用。如果一個目標(biāo)有多個觀察者,那么目標(biāo)也依賴所有觀察者,從而目標(biāo)對象無法獨(dú)立復(fù)用。如何消除目標(biāo)和觀察者之間的互相依賴呢?觀察者模式幫助我們解決這個問題。
觀察者模式把目標(biāo)對觀察者的依賴進(jìn)行抽象:使目標(biāo)只知道自己有若干觀察者,但不知道這些觀察者具體是誰,可能有多少個;當(dāng)目標(biāo)狀態(tài)改變時只要給這些觀察者一個通知,不必作更多的事情。這樣目標(biāo)對觀察者的依賴就達(dá)到了抽象和最小,而目標(biāo)對具體觀察者的依賴被解除了。
類圖如下:


Subject 對象保存一個Observer引用的列表,當(dāng)我們讓一個ConcreteObserver對象觀察Subject對象時,調(diào)用后者的Attach()方法,將前者的引用加入該列表中。當(dāng)Subject對象狀態(tài)改變時,它調(diào)用自身的Notify方法,該方法調(diào)用列表中每一個Observer的 Update()方法。一個ConcreteObserver只要重定義Update()就能收到通知,作為對通知的響應(yīng),Update()調(diào)用 Subject對象的getStatus()獲取數(shù)據(jù),然后更新自身。當(dāng)不需要繼續(xù)觀察時,ConcreteObserver對象調(diào)用Subject對象的Detach()方法,其引用被從列表中移除。
解除目標(biāo)對具體觀察者的依賴以后,很容易增加新的具體觀察者,因?yàn)椴皇芤蕾嚨姆矫婢涂梢宰杂勺兓欢繕?biāo)也可以獨(dú)立地復(fù)用,因?yàn)闊o所依賴的方面就可以不受影響。
以上主要考慮了一個目標(biāo)有多個觀察者的情況,我們設(shè)法解除了目標(biāo)對具體觀察者的依賴,使具體觀察者的種類和數(shù)目容易改變。有時候一個觀察者觀察多個目標(biāo)也是有意義的,在前面的類圖中,觀察者對具體目標(biāo)的依賴仍然存在,因此無法適應(yīng)目標(biāo)方面的變化。怎樣抽象這種依賴呢?使觀察者只知道若干個目標(biāo)會向自己發(fā)出通知,而不知道這些目標(biāo)具體是誰,可能有多少個;在目標(biāo)向觀察者發(fā)送通知時,將一個自身的引用作為參數(shù),然后觀察者調(diào)用其抽象方法就可以獲得目標(biāo)狀態(tài)。這就使得觀察者對目標(biāo)的依賴是抽象的,觀察者對具體目標(biāo)的依賴被解除了。
類圖如下:

參考資料:
1.《設(shè)計(jì)模式-可復(fù)用面向?qū)ο筌浖幕A(chǔ)》/Erich Gamma等著,李英軍等譯 機(jī)械工業(yè)出版社