1)  观察者模式定义

略,各种设计模式的书上都有定义。

2)  观察者模式一般实现

观察者模式一般实现,都是“被观察者”保存一个“观察者”的列表,循环这个列表来通知“观察者”。代码,其中使用了boost的智能指针shared_ptr:

  1. #include <iostream>
  2. #include <list>
  3. #include "boost/shared_ptr.hpp"
  4. using namespace std;
  5. using namespace boost;
  6. //观察者基类
  7. class Observer
  8. {
  9. public:
  10. //处理被观察者的通知
  11. virtual void update(const int i) = 0;
  12. };
  13. //观察者类A
  14. class ObserverA : public Observer
  15. {
  16. public:
  17. virtual void update(const int i)
  18. {
  19. cout << "ObserverA.update(" << i << ")" << endl;
  20. }
  21. };
  22. //观察者类B
  23. class ObserverB : public Observer
  24. {
  25. public:
  26. virtual void update(const int i)
  27. {
  28. cout << "ObserverB.update(" << i << ")" << endl;
  29. }
  30. };
  31. //被观察者基类
  32. class Subject
  33. {
  34. public:
  35. virtual ~Subject()
  36. {
  37. }
  38. //注册观察者
  39. virtual void reg(const shared_ptr<Observer>& pObs) = 0;
  40. //取消注册观察者
  41. virtual void unreg(const shared_ptr<Observer>& pObs) = 0;
  42. //通知观察者
  43. virtual void notify(const int i) = 0;
  44. };
  45. //被观察者类A
  46. class SubjectA : public Subject
  47. {
  48. public:
  49. virtual void reg(const shared_ptr<Observer>& pObs)
  50. {
  51. m_lstObs.push_back(pObs);
  52. }
  53. virtual void unreg(const shared_ptr<Observer>& pObs)
  54. {
  55. for (list<shared_ptr<Observer> >::iterator it = m_lstObs.begin();
  56. it != m_lstObs.end();)
  57. {
  58. if (pObs == *it) m_lstObs.erase(it++);
  59. else ++it;
  60. }
  61. }
  62. virtual void notify(const int i)
  63. {
  64. for (list<shared_ptr<Observer> >::iterator it = m_lstObs.begin();
  65. it != m_lstObs.end(); ++it)
  66. {
  67. (*it)->update(i);
  68. }
  69. }
  70. protected:
  71. //观察者类别
  72. list<shared_ptr<Observer> > m_lstObs;
  73. };
  74. int main()
  75. {
  76. //被观察者pSubA
  77. shared_ptr<Subject> pSubA(new SubjectA);
  78. //观察者pObsA
  79. shared_ptr<Observer> pObsA(new ObserverA);
  80. //观察者pObsB
  81. shared_ptr<Observer> pObsB(new ObserverB);
  82. //注册pObsA
  83. pSubA->reg(pObsA);
  84. //注册pObsB
  85. pSubA->reg(pObsB);
  86. //通知
  87. pSubA->notify(1);                //显示
  88. //ObserverA.update(1)
  89. //ObserverB.update(1)
  90. //取消pObsA的注册
  91. pSubA->unreg(pObsA);
  92. //通知
  93. pSubA->notify(2);                //显示
  94. //ObserverB.update(2)
  95. return 0;
  96. }

3)  观察者模式Boost.Signals实现

Boost.Signals是基于函数指针或函数对象来实现观察者模式的。其中Boost.Signals2是线程安全版本。代码:

  1. #include <iostream>
  2. #include <list>
  3. #include "boost/shared_ptr.hpp"
  4. #include "boost/signal.hpp"
  5. using namespace std;
  6. using namespace boost;
  7. //使用函数对象来实现
  8. //观察者类A
  9. class ObserverA
  10. {
  11. public:
  12. void operator()(const int i)
  13. {
  14. cout << "ObserverA(" << i << ")" << endl;
  15. }
  16. };
  17. //观察者类B
  18. class ObserverB
  19. {
  20. public:
  21. void operator()(const int i)
  22. {
  23. cout << "ObserverB(" << i << ")" << endl;
  24. }
  25. };
  26. int main()
  27. {
  28. //被观察者sig
  29. signal<void (const int)> sig;
  30. //注册观察者ObserverA
  31. signals::connection connA = sig.connect(ObserverA());
  32. //注册观察者ObserverB
  33. sig.connect(ObserverB());
  34. //通知
  35. sig(1);                 //显示
  36. //ObserverA(1)
  37. //ObserverB(1)
  38. //取消obsA的注册
  39. connA.disconnect();
  40. //通知
  41. sig(2);                 //显示
  42. //ObserverB(2)
  43. return 0;
  44. }

Boost.Signals还支持被观察者分组,可以对整个组进行“通知”、取消注册。另外,Boost.Signals需要编译才能使用。

观察者模式与Boost.Signals的更多相关文章

  1. boost::signals::signal的使用方法

    吃力的讲完boost::signals的ppt.然后接着就是做练习题. 通过讲ppt,发现有一句话说的真好:你自己知道是一回事.你能给别人讲明确又是另外一回事.真的有些东西你自己理解,可是用语言去非常 ...

  2. vs2013 boost signals

    #include "stdafx.h" #include <boost/signals2/signal.hpp> #include <iostream> u ...

  3. Linux上安装使用boost入门指导

    Data Mining Linux上安装使用boost入门指导 获得boost boost分布 只需要头文件的库 使用boost建立一个简单的程序 准备使用boost二进制文件库 把你的程序链接到bo ...

  4. Windows下如何使用BOOST C++库 .

    Windows下如何使用BOOST C++库 我采用的是VC8.0和boost_1_35_0.自己重新编译boost当然可以,但是我使用了 http://www.boostpro.com/produc ...

  5. 以boost::function和boost:bind取代虚函数

    转自:http://blog.csdn.net/Solstice/archive/2008/10/13/3066268.aspx 这是一篇比较情绪化的blog,中心思想是“继承就像一条贼船,上去就下不 ...

  6. Win7下Boost库的安装

    Boost库是C++领域公认的经过千锤百炼的知名C++类库,涉及编程中的方方面面,简单记录一下使用时的安装过程 1.boost库的下载 boost库官网主页:www.boost.org 2.安装 将下 ...

  7. VS2008下直接安装使用Boost库1.46.1版本号

    Boost库是一个可移植.提供源码的C++库,作为标准库的后备,是C++标准化进程的发动机之中的一个. Boost库由C++标准委员会库工作组成员发起,当中有些内容有望成为下一代C++标准库内容.在C ...

  8. VS2008下直接安装使用Boost库1.46.1版本

    Boost库是一个可移植.提供源代码的C++库,作为标准库的后备,是C++标准化进程的发动机之一. Boost库由C++标准委员会库工作组成员发起,其中有些内容有望成为下一代C++标准库内容.在C++ ...

  9. boost事件处理

    尽管这个库的名字乍一看好象有点误导,但实际上并不是如此. Boost.Signals 所实现的模式被命名为 '信号至插槽' (signal to slot).它基于下面概念:当相应的信号被发出时.相关 ...

随机推荐

  1. 用JS来计算百钱买百鸡

    怎样用一百块买一百只鸡?已知公鸡5块一只,母鸡3块一只,小鸡一块钱3只: 需要用到for循环嵌套,并且通过优化代码,可以加快运行效率. <!DOCTYPE html> <html l ...

  2. JQuery或JavaScript获取网页的宽度、高等

    最近多次使用JQery或JavaScript获取网页的宽度或者高度,在网上搜索N久之后发现很多都是粘贴上去并没有详细的介绍,这里我将会对经常使用的一些获取页面宽高的属性,方法做详细的介绍,以便能够更加 ...

  3. MySQL 一些小知识

    1. 关于多表查询 我的理解:由于MySQL多表查询时表之间的连接是笛卡尔积的方式,所以尽量少使用多表查询,如果使用则使用嵌套语句 例:说明: `tb_notice_message` 表数量百万级别以 ...

  4. vs2012快捷键失效解决办法

    快速解决vs开发工具快捷键失效,看图

  5. Controller返回值类型ActionResult

    在mvc中所有的controller类都必须使用"Controller"后缀来命名 并且对Action也有一定的要求: 必须是一个public方法 必须是实例方法 没有标志NonA ...

  6. SQL Server链接MySQL实践

    最近在访问多数据库的时候进行了SQLServer链接MySQL数据的实践,现总结如下: 一.  安装mysql-connector-odbc驱动: 1. 在SQL Server服务器的机器上安装mys ...

  7. c#字符串方法

    作者: 常浩 staticvoid Main(string[] args) { string s =""; //(1)字符访问(下标访问s[i]) s ="ABCD&qu ...

  8. linux install zh_CN(ubuntu)

    cd /usr/share/locales sudo ./install-language-pack zh_CN

  9. gitolite随记

    1.git clone源码 git clone git://github.com/sitaramc/gitolite 2.安装 gitolite/install -ln 3.建立git仓库 gitol ...

  10. hdu 2199

    Problem Description Now,given the equation 8*x^4 + 7*x^3 + 2*x^2 + 3*x + 6 == Y,can you find its sol ...