• <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>

            我的玻璃盒子

            (轉載)觀察者 (Subject/Observer) 模式實現

            原文鏈接:http://www.shnenglu.com/eXile/archive/2007/09/16/32297.html

              以前我曾經實現過觀察者模式(signal / slot )。有位朋友不以為然,也把他的實現發給我。這是用純OO的方式實現的,沒有使用模板,不依賴于其它庫。應該是仿Java或C#接口。設計得不錯,具有以下特點:
              1)當Subject或Observer 銷毀時,連接自動斷開(注冊自動取消),當然,這也是實現該模式的重點。
              2)考慮了以下因素:一個Subject是否允許多個Observer觀察?一個Observer是否允許同時觀察多個Subject? 由此可分為一對一,一對多,多對多,一般的GUI庫中都是后兩種情況,但自己寫的程序中卻第一種情況居多。所以他出于效率,設計了SimpleSubject和SimpleObserver。
              不足之處如下:
              1)Observer::update()只能帶無類型的參數(好象也沒有更好的辦法)。
              2)由于C++中沒有匿名類,所以使用起來并沒有Java中方便。于是我又添加了一個ObserverAdapter。
              使用舉例:


            class A : public Subject
            {
                    int value;
            public:
               void setValue(int v)
               {
                   value = v;
                   notify(&value);
               }      
            };

            class B : public Observer
            {
            public:
               void update(void* arg)
               {
                   printf("A changed:%d", *(int*)arg);
               }      
            };

            void f(A* a, B* b)
            {
                a->connect(b);
                a->setValue(1);
            }

            使用ObserverAdapter ,則變成了以下情況:

            class B
            {
               ObserverAdapter<B> observerOfA;
            public:
               B(A* a) : observerOfA(this, &B::valueChanged)
               {
                   a->connect(&observerOfA);
               }

               void valueChanged(void* arg)
               {
                   printf("A changed:%d", *(int*)arg);
               }      
            };

              代碼分為兩個文件:IObserver.h 和 Observer.h,就貼在下面吧:
            IObserver.h  : 接口定義


            #pragma once

            class ISubject;

            class IObserver
            {
            public:
                IObserver()  {}
                virtual ~IObserver() {};

                virtual void update(void*) = 0;

            protected:
                virtual void addSubject(ISubject* ) = 0;
                virtual void removeSubject(ISubject* ) = 0;

                friend class ISubject;

            private:
                IObserver(IObserver const&);
                IObserver& operator= (IObserver const&);
            };

            class ISubject
            {
            public:
                ISubject() {}
                virtual ~ISubject() {};

                virtual void connect(IObserver*) = 0;
                virtual void disconnect(IObserver*) = 0;
                virtual bool isConnected(IObserver*) const= 0;
                virtual void notify(void*) = 0;

            protected:
                void addObserver(IObserver* observer);
                void removeObserver(IObserver* observer);

            private:
                ISubject(ISubject const&);
                ISubject& operator= (ISubject const&);
            };

            Observer.h  : 具體實現


            #pragma once

            #include <cassert>
            #include <set>

            #include "IObserver.h"

            //-------------------------------------------------------------------

            inline void ISubject::addObserver(IObserver* observer)
            {
                observer->addSubject(this);
            }

            inline void ISubject::removeObserver(IObserver* observer)
            {
                observer->removeSubject(this);
            }

            //-------------------------------------------------------------------

            class SimpleSubject : public ISubject
            {
            public:
                SimpleSubject() : m_observer(0)
                {
                }

                ~SimpleSubject()
                {
            if (m_observer) removeObserver(m_observer);
                }

                virtual void connect(IObserver* observer)
                {
                    assert(observer);
            if (m_observer)
                        removeObserver(m_observer);
                    addObserver(observer);
                    m_observer = observer;
                }

                virtual void disconnect(IObserver* observer)
                {
                    assert(observer && observer == m_observer);
                    removeObserver(m_observer);
                    m_observer = 0;
                }

                virtual bool isConnected(IObserver* observer) const
                {
                    return observer == m_observer;
                }

                virtual void notify(void* arg)
                {
            if (m_observer) m_observer->update(arg);
                }

            private:
                IObserver*  m_observer;

            };

            //-------------------------------------------------------------------

            class Subject : public ISubject
            {
            public:
                Subject() : m_observers()
                {
                }

                ~Subject()
                {
                    std::set<IObserver*>::iterator
                        it = m_observers.begin(),
                        e  = m_observers.end();
            for (; it != e; ++it)
                    {
                        removeObserver(*it);
                    }
                }

                virtual void connect(IObserver* observer)
                {
                    assert(observer);
                    addObserver(observer);
                    m_observers.insert(observer);
                }

                virtual void disconnect(IObserver* observer)
                {
                    assert(observer);
                    removeObserver(observer);
                    m_observers.erase(observer);
                }

                virtual bool isConnected(IObserver* observer) const
                {
                    return m_observers.find(observer) != m_observers.end();
                }

                virtual void notify(void* arg)
                {
                    std::set<IObserver*>::iterator
                        it = m_observers.begin(),
                        e  = m_observers.end();
            while (it != e)
                    {
                        (*it++)->update(arg); // observer can be disconnected in update()
                    }
                }

            private:
                std::set<IObserver*>  m_observers;

            };

            //-------------------------------------------------------------------

            class SimpleObserver  : public IObserver
            {
            public:
                SimpleObserver() : m_subject(0)
                {
                }

                ~SimpleObserver()
                {
            if (m_subject) m_subject->disconnect(this);
                }

                ISubject* getSubject() const
                {
                    return m_subject;
                }

            private:
                virtual void addSubject(ISubject* subject)
                {
            if (m_subject) m_subject->disconnect(this);
                    m_subject = subject;
                }

                virtual void removeSubject(ISubject* subject)
                {
                    assert(subject == m_subject);
                    m_subject = 0;
                }

            private:
                ISubject*   m_subject;
            };


            //-------------------------------------------------------------------

            class Observer : public IObserver
            {
            public:
                Observer() : m_subjects()
                {
                }

                ~Observer()
                {
                    std::set<ISubject*>::iterator
                        it = m_subjects.begin(),
                        e  = m_subjects.end();
            while (it != e)
                    {
                        (*it++)->disconnect(this);
                    }
                }

            private:
                virtual void addSubject(ISubject* subject)
                {
                    assert(subject);
                    m_subjects.insert(subject);
                }

                virtual void removeSubject(ISubject* subject)
                {
                    assert(subject);
                    m_subjects.erase(subject);
                }

            private:
                std::set<ISubject*>   m_subjects;
            };

            //-------------------------------------------------------------------

            template <class T, class Base = SimpleObserver>
            class ObserverAdapter  : public Base
            {
            public:
                ObserverAdapter(T* t, void (T::*f)(void*))
                 : m_obj(t), m_func(f)
                {
                }

                virtual void update(void* arg)
                {
                    (m_obj->*m_func)(arg);
                }

            private:
                T*  m_obj;
                void (T::*m_func)(void*);
            };

            //-------------------------------------------------------------------

            posted on 2008-01-29 16:17 深藍色系統 閱讀(403) 評論(1)  編輯 收藏 引用 所屬分類: Skills

            評論

            # re: (轉載)觀察者 (Subject/Observer) 模式實現 2009-07-27 16:19 欲三更

            要想不攜帶void *參數,還是得用模板。  回復  更多評論   

            導航

            <2009年7月>
            2829301234
            567891011
            12131415161718
            19202122232425
            2627282930311
            2345678

            統計

            常用鏈接

            留言簿(75)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            狠狠色丁香久久婷婷综合_中| 激情久久久久久久久久| 欧美日韩精品久久久免费观看| 亚洲国产精品无码久久九九| 久久久黄色大片| 精品久久久久久无码人妻蜜桃 | 久久精品国产亚洲AV影院| 中文国产成人精品久久不卡| 伊人久久大香线蕉影院95| 亚洲午夜福利精品久久 | 久久精品国产99久久久香蕉| 狠狠综合久久综合88亚洲| 国产精品美女久久久久av爽 | 亚洲人AV永久一区二区三区久久 | 国内精品伊人久久久久网站| 97久久国产露脸精品国产| 91秦先生久久久久久久| 亚洲精品无码久久久久去q| 久久99亚洲综合精品首页| .精品久久久麻豆国产精品| 久久亚洲中文字幕精品一区| 久久一区二区免费播放| 国产成人久久精品区一区二区| 精品一二三区久久aaa片| 日韩亚洲国产综合久久久| 欧美伊香蕉久久综合类网站| 久久久久AV综合网成人| 国产69精品久久久久久人妻精品| 久久久久国产精品麻豆AR影院 | 97久久国产综合精品女不卡| 久久久久久毛片免费看| 99久久国产主播综合精品| 久久精品国内一区二区三区| 久久久久人妻精品一区二区三区| 伊人久久大香线蕉综合热线| 久久久网中文字幕| 久久精品99无色码中文字幕| 国产高潮久久免费观看| 久久久久亚洲爆乳少妇无| 国产精品va久久久久久久| 狠狠色综合网站久久久久久久 |