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. cookie 和 session

    会话(Session)跟踪是Web程序中常用的技术,用来跟踪用户的整个会话.常用的会话跟踪技术是Cookie与Session.Cookie通过在客户端记录信息确定用户身份,Session通过在服务器端 ...

  2. MS-SQL数据库备份方法

    一.手动备份 打开企业管理器 --> 右键点击需要备份的数据库 --> 所有任务 --> 备份数据库 或者: 查询分析器: use master  backup database 数 ...

  3. easyui treeJson 带层数

    public string GetTreeNav(int ID,int Num) { StringBuilder sb = new StringBuilder(); sb.Append("[ ...

  4. Android学习之Activity之间的数据传递

    Activity与Activity之间很多情况下都需要进行数据的传递,下面就用几个简单的例子来看一下. (一).一个Activity启动另一个Activity并将数据传递到这个Activity当中 思 ...

  5. oracle 自动添加序号列 排序

    select      HSL.sortno,                    HSL.B,                    HSL.A,                    row_n ...

  6. Struts2+Spring3+Hibernate3+Maven构建(基于Eclipse)

    长时间不做后台了,整理一下资料,以便翻阅. Eclipse.JDK安装略…… Maven下载地址:http://maven.apache.org/download.cgi 版本比较新的Eclipse基 ...

  7. [转载]VIM命令合集

    Vim命令合集 http://www.cnblogs.com/softwaretesting/archive/2011/07/12/2104435.html 命令历史 以:和/开头的命令都有历史纪录, ...

  8. Linux下运行C++程序出现"段错误(核心已转储)"的原因

    今天写程序出现了“段错误(核心已转储)"的问题,查了一下资料,加上自己的实践,总结了以下几个方面的原因. 1.内存访问出错  这类问题的典型代表就是数组越界. 2.非法内存访问 出现这类问题 ...

  9. Maven基础教程

    更多内容请参考官方文档:http://maven.apache.org/guides/index.html 官方文档很详细,基本上可以查找到一切相关的内容. 另外,快速入门可参考视频:孔浩的maven ...

  10. 使用ganymed-ssh2-build通过ssh获得远程服务器参数

    1.项目中需要检测到几台远程服务器的参数,差了很多资料,决定用的这个 2.jar包:ganymed-ssh2-build210.jar 3.原理:向远程linux服务器发送脚本命令,得到该台服务器的信 ...