• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            設計模式之觀察者模式

            一、什么是觀察者模式

                 Observer模式也叫觀察者模式,是由GoF提出的23種軟件設計模式的一種。Observer模式是行為模式之一,它的作用是當一個對象的狀態發生變化時,能夠自動通知其他關聯對象,自動刷新對象狀態。

              觀察者模式(Observer)完美的將觀察者和被觀察的對象分離開。舉個例子,用戶界面可以作為一個觀察者,業務數據是被觀察者,用戶界面觀察業務數據的變化,發現數據變化后,就顯示在界面上。面向對象設計的一個原則是:系統中的每個類將重點放在某一個功能上,而不是其他方面。一個對象只做一件事情,并且將他做好。觀察者模式在模塊之間劃定了清晰的界限,提高了應用程序的可維護性和重用性。

             

              觀察者模式有很多實現方式,從根本上說,該模式必須包含兩個角色:觀察者和被觀察對象。在剛才的例子中,業務數據是被觀察對象,用戶界面是觀察者。觀察者和被觀察者之間存在“觀察”的邏輯關聯,當被觀察者發生改變的時候,觀察者就會觀察到這樣的變化,并且做出相應的響應。如果在用戶界面、業務數據之間使用這樣的觀察過程,可以確保界面和數據之間劃清界限,假定應用程序的需求發生變化,需要修改界面的表現,只需要重新構建一個用戶界面,業務數據不需要發生變化。

             

              “觀察”不是“直接調用”

              實現觀察者模式的時候要注意,觀察者和被觀察對象之間的互動關系不能體現成類之間的直接調用,否則就將使觀察者和被觀察對象之間緊密的耦合起來,從根本上違反面向對象的設計的原則。無論是觀察者“觀察”觀察對象,還是被觀察者將自己的改變“通知”觀察者,都不應該直接調用。

             

              實現觀察者模式的形式

              實現觀察者模式有很多形式,比較直觀的一種是使用一種“注冊——通知——撤銷注冊”的形式。

             

               實現觀察者模式例子

                   下面是C++的實現,在C++實現中,C++中沒有接口的概念,但是可以用抽象類類代替Java或C#中的接口,在C++中抽象類中從派生類中抽象出來的函數(方法),必須定義成純虛函數,這樣在后面的使用中才可以通過基類的指針來訪問這些函數,面向對象的語言中有個特點,多態只能訪問兩者中共有的部分。

              1
              2#include "stdafx.h"
              3#include <string>
              4#include <vector>
              5#include<list>
              6#include<iostream>
              7using namespace std;
              8
              9class Observer;
             10class Subject
             11{
             12public:
             13    //注冊
             14     virtual void attach(Observer *o)=0;
             15     //撤銷
             16     virtual void dettach(Observer *o)=0;
             17     //發送消息,更新觀察者數據
             18     virtual void change()=0;
             19     //更新數據
             20     virtual void setWeather(string str)=0;
             21     //獲取數據
             22     virtual string getWeather()=0;
             23}
            ;
             24
             25class Observer
             26{
             27public:
             28    //獲得觀察者的名字,用于后面dettach撤銷操作
             29     virtual string getName()=0;
             30     //更新數據,用于被觀察者調用
             31     virtual void update(Subject *s)=0;
             32}
            ;
             33
             34//被觀察者
             35class Earth: public Subject
             36{
             37private:
             38    //數據
             39     string weather;
             40     //存放觀察者的list,因為Observer是個抽象類,
             41     //所以不能聲明成 list<Observer >* l
             42     list<Observer* >* l;//指針 
             43public:
             44    //初始化存放觀察者的容器
             45     Earth()
             46     {
             47        l = new list<Observer*>;
             48     }
            ;
             49     //別忘了釋放資源
             50    ~Earth()
             51    {
             52        delete l;
             53    }
            ;
             54    //注冊觀察者
             55     void attach(Observer *o)
             56     {
             57        this->l->push_back(o);
             58     }
            ;
             59    
             60     //注銷觀察者
             61      void dettach(Observer *o)
             62     {
             63         for(list<Observer*>::iterator it=l->begin();it!=l->end();++it)
             64         {
             65             //通過string來進行查找,其他通過指針怎么判斷相等比較困難
             66             if( (*it)->getName() == o->getName())
             67             {
             68                 l->remove(*it);
             69                 break;
             70             }

             71         }

             72     }
            ;
             73
             74    //通知并更新觀察者
             75     void change()
             76     {
             77          for(list<Observer*>::iterator it=l->begin();it!=l->end();++it)
             78          {
             79            (*it)->update(this);
             80          }

             81     }
            ;
             82
             83     //更新數據
             84     void setWeather(string str)
             85     {
             86          this->weather=str;
             87          change();
             88     }
            ;
             89
             90     string getWeather()
             91     {
             92         return this->weather;
             93     }
            ;
             94
             95}
            ;
             96
             97class Satellite:public Observer
             98{
             99private:
            100     string name;
            101public:
            102     Satellite(string str)
            103     {
            104          name=str;
            105     }

            106     string getName()
            107     {
            108         return name;
            109     }
            ;
            110     //更新數據
            111     void update(Subject *s)
            112     {
            113         cout<<this->getName()+" "+s->getWeather()<<endl;
            114     }

            115}
            ;
            116
            117
            118
            119int _tmain(int argc, _TCHAR* argv[])
            120{
            121     Earth e;
            122     Satellite *s1 = new Satellite("風云一號");
            123     Satellite *s2 = new Satellite("風云二號");
            124     Satellite *s3 = new Satellite("風云三號");
            125     Satellite *s4 = new Satellite("風云四號");
            126     e.attach(s1);
            127     e.attach(s2);
            128     e.attach(s3);
            129     e.attach(s4);
            130     e.setWeather("fine");
            131     e.setWeather("rain");
            132     //注銷s3
            133     e.dettach(s3);
            134     e.setWeather("fine");
            135     e.setWeather("rain");
            136
            137
            138    return 0;
            139}

            140
            141

            以上文字參照http://blog.csdn.net/preciousboy/article/details/6230467,對其代碼進行修改在vs2008中編譯調試通過。
            結果運行為:


            這里要注意的幾個問題是?
            1.被觀察者怎么去通知觀察者數據已經更新?--本例子中是通過void change()函數來實現的;
            2.在此模式中你可以從被觀察者處push或者pull數據,我們認為push的方式是正確的。
            3.觀察者模式定義了一對多的關系
            4.有多個觀察者時不依賴于特定的通知次序。
            5.java的MVC模式就采用此模式實現。

            posted on 2012-05-01 11:47 sheng 閱讀(644) 評論(0)  編輯 收藏 引用

            導航

            <2011年12月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            統計

            常用鏈接

            留言簿(1)

            隨筆檔案

            收藏夾

            同行

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            久久综合九色综合精品| 亚洲国产成人久久综合区| 国产人久久人人人人爽| 青青草原综合久久大伊人精品| 久久影视综合亚洲| 国产亚洲色婷婷久久99精品| 久久av免费天堂小草播放| 亚洲AV无码久久精品成人| 日本精品久久久中文字幕| 精品国产日韩久久亚洲| 久久电影网一区| 精产国品久久一二三产区区别 | 99久久精品午夜一区二区| 久久久久国产精品三级网| 亚洲AV无码1区2区久久| 久久久久久久国产免费看| 久久精品一区二区三区不卡| 亚洲美日韩Av中文字幕无码久久久妻妇 | 久久人做人爽一区二区三区| 国产91色综合久久免费分享| 99久久国产亚洲综合精品| 亚洲国产成人久久综合一| 欧美牲交A欧牲交aⅴ久久| 中文字幕无码久久人妻| 国产综合精品久久亚洲| 久久精品aⅴ无码中文字字幕重口| 亚洲精品乱码久久久久久不卡| 久久精品国产99国产精偷| 久久天天躁狠狠躁夜夜avapp| 久久久久久久久波多野高潮| 三级韩国一区久久二区综合| 久久人人爽人人爽人人片AV麻豆| 久久精品国产91久久综合麻豆自制 | 一本色道久久99一综合| 波多野结衣久久一区二区| 亚洲国产成人久久综合一区77| 国内精品久久久久影院网站| 国产精品99久久久久久猫咪| 午夜不卡888久久| 久久91这里精品国产2020| 国産精品久久久久久久|