• <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 深藍色系統 閱讀(416) 評論(1)  編輯 收藏 引用 所屬分類: Skills

            評論

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

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

            導航

            <2008年1月>
            303112345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            統計

            常用鏈接

            留言簿(75)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            欧美激情精品久久久久| .精品久久久麻豆国产精品| 亚洲国产精品综合久久网络 | 国产一区二区三区久久| 国内精品久久久久国产盗摄| 久久久久99这里有精品10| 欧美噜噜久久久XXX| 亚洲精品乱码久久久久久不卡| 亚洲va久久久噜噜噜久久狠狠| 91精品国产高清久久久久久国产嫩草 | 嫩草影院久久国产精品| 中文字幕久久精品无码| 久久久久久久久久久免费精品| 久久久婷婷五月亚洲97号色| 无码人妻久久一区二区三区蜜桃| 77777亚洲午夜久久多喷| 久久精品无码一区二区WWW| 久久996热精品xxxx| 久久99国产精品二区不卡| 色婷婷久久综合中文久久蜜桃av | 久久久久免费视频| 久久91精品国产91久久小草| 久久综合亚洲鲁鲁五月天| 欧洲国产伦久久久久久久| 国产视频久久| 国产综合精品久久亚洲| 日本精品久久久久中文字幕8| 狠狠狠色丁香婷婷综合久久俺| 亚洲va久久久噜噜噜久久男同| 一本一本久久a久久综合精品蜜桃| 欧美午夜A∨大片久久| 久久伊人色| 麻豆久久久9性大片| 国产人久久人人人人爽| 青青草原综合久久大伊人导航| 91久久精品无码一区二区毛片| 性高湖久久久久久久久| 亚洲国产成人精品无码久久久久久综合| 久久国产欧美日韩精品| 无码人妻久久一区二区三区免费丨| 日韩欧美亚洲综合久久|