C++ 观察者模式样例
  #include <iostream>
  #include <set>
  #include <string>
  using namespace std;
  /////////////////////抽象模式定义
  class CObservable;
  //观察者,纯虚基类
  class CObserver
  {
  public:
  CObserver::CObserver(){};
  virtual CObserver::~CObserver(){};
  //当被观察的目标发生变化时,通知调用该方法
  //来自被观察者pObs, 扩展參数为pArg
  virtual void Update(CObservable* pObs, void* pArg = NULL) = 0;
  };
  //被观察者,即Subject
  class CObservable
  {
  public:
  CObservable() : m_bChanged(false) {};
  virtual ~CObservable() {};
  //注冊观察者
  void Attach(CObserver* pObs);
  //注销观察者
  void Detach(CObserver* pObs);
  //注销全部观察者
  void DetachAll();
  //若状态变化,则遍历观察者,逐个通知更新
  void Notify(void* pArg = NULL);
  //測试目标状态是否变化
  bool HasChanged();
  //获取观察者数量
  int GetObserversCount();
  protected:
  //设置状态变化!!!必须继承CObservable才干设置目标状态
  void SetChanged();
  //初始化目标为未变化状态
  void ClearChanged();
  private:
  bool m_bChanged; //状态
  set<CObserver*> m_setObs; //set保证目标唯一性
  };
  /////////////////////抽象模式实现
  void CObservable::Attach(CObserver* pObs)
  {
  if (!pObs) return;
  m_setObs.insert(pObs);
  }
  void CObservable::Detach(CObserver* pObs)
  {
  if (!pObs) return;
  m_setObs.erase(pObs);
  }
  void CObservable::DetachAll()
  {
  m_setObs.clear();
  }
  void CObservable::SetChanged()
  {
  m_bChanged = true;
  }
  void CObservable::ClearChanged()
  {
  m_bChanged = false;
  }
  bool CObservable::HasChanged()
  {
  return m_bChanged;
  }
  int CObservable::GetObserversCount()
  {
  return m_setObs.size();
  }
  void CObservable::Notify(void* pArg /* = NULL */)
  {
  if (!HasChanged()) return;
  cout << "notify observers…" << endl;
  ClearChanged();
  set<CObserver*>::iterator itr = m_setObs.begin();
  for (; itr != m_setObs.end(); itr++)
  {
  (*itr)->Update(this, pArg);
  }
  }
  /////////////////////详细应用类定义和实现
  //bloger是公布者,即被观察者(subject)
  class CBloger : public CObservable
  {
  public:
  void Publish(const string &strContent)
  {
  cout << "bloger publish, content: " << strContent << endl;
  SetChanged();
  Notify(const_cast<char*>(strContent.c_str()));
  }
  };
  //portal是公布者,即被观察者(subject)
  class CPortal : public CObservable
  {
  public:
  void Publish(const string &strContent)
  {
  cout << "portal publish, content: " << strContent << endl;
  SetChanged();
  Notify(const_cast<char*>(strContent.c_str()));
  }
  };
  //RSS阅读器,观察者
  class CRSSReader : public CObserver
  {
  public:
  CRSSReader(const string &strName) : m_strName(strName){}
  virtual void Update(CObservable* pObs, void* pArg = NULL)
  {
  char* pContent = static_cast<char*>(pArg);
  //观察多个目标
  if (dynamic_cast<CBloger*>(pObs))
  {
  cout << m_strName << " updated from bloger, content: " << pContent << endl;
  }
  else if (dynamic_cast<CPortal*>(pObs))
  {
  cout << m_strName << " updated from portal, content: " << pContent << endl;
  }
  }
  private:
  string m_strName;
  };
  //Mail阅读器,观察者
  class CMailReader : public CObserver
  {
  public:
  CMailReader(const string &strName) : m_strName(strName){}
  virtual void Update(CObservable* pObs, void* pArg = NULL)
  {
  char* pContent = static_cast<char*>(pArg);
  if (dynamic_cast<CBloger*>(pObs))
  {
  cout << m_strName << " updated from bloger, content: " << pContent << endl;
  }
  if (dynamic_cast<CPortal*>(pObs))
  {
  cout << m_strName << " updated from portal, content: " << pContent << endl;
  }
  }
  private:
  string m_strName;
  };
  /////////////////Main
  int main()
  {
  //目标(被观察者)
  CBloger* pBloger = new CBloger();
  CPortal* pPortal = new CPortal();
  //观察者. 一个观察者能够观察多个目标
  CRSSReader* pRssReader = new CRSSReader("rss reader");
  CMailReader* pMailReader = new CMailReader("mail reader");
  pBloger->Attach(pRssReader); //bloger注冊观察者
  pBloger->Attach(pMailReader); //bloger注冊观察者
  pPortal->Attach(pRssReader); //portal注冊观察者
  pPortal->Attach(pMailReader); //portal注冊观察者
  //博客公布信息
  pBloger->Publish("博客分享设计模式");
  cout << endl;
  //门户公布信息
  pPortal->Publish("门户分享设计模式");
  cout << "\nportal detached mail reader" << endl;
  pPortal->Detach(pMailReader);
  cout << "portal observers count: " << pPortal->GetObserversCount() << endl << endl;
  pPortal->Publish("门户分享设计模式");
  system("pause");
  return 0;
  }

C++ 观察者模式样例的更多相关文章

  1. java设计模式演示样例

    创建模式 1.工厂方法模式(Factory Method)  将程序中创建对象的操作,单独出来处理,创建一个产品的工厂接口,把实际的工作转移到详细的子类.大大提高了系统扩展的柔性,接口的抽象化处理给相 ...

  2. C++的性能C#的产能?! - .Net Native 系列《三》:.NET Native部署测试方案及样例

    之前一文<c++的性能, c#的产能?!鱼和熊掌可以兼得,.NET NATIVE初窥> 获得很多朋友支持和鼓励,也更让我坚定做这项技术的推广者,希望能让更多的朋友了解这项技术,于是先从官方 ...

  3. MarkDown+LaTex 数学内容编辑样例收集

    $\color{green}{MarkDown+LaTex 数学内容编辑样例收集}$ 1.大小标题的居中,大小,颜色 [例1] $\color{Blue}{一元二次方程根的分布}$ $\color{R ...

  4. 33个超级有用必须要收藏的PHP代码样例

    作为一个正常的程序员,会好几种语言是十分正常的,相信大部分程序员也都会编写几句PHP程序,如果是WEB程序员,PHP一定是必备的,即使你没用开发过大型软件项目,也一定多少了解它的语法. 在PHP的流行 ...

  5. 关于peersim样例配置文件的超详细解读(新手勿喷)

    相信很多兄弟一开始接触peersim,对配置文件还是有点不适应,我看了好久的样例的配置文件,一层层去找对应的文件的方法,终于好像悟懂了一点,记下来以后回顾. 贴上代码,一点点分析. 首先要说下所谓的配 ...

  6. hdu 1003 MAX SUM 简单的dp,测试样例之间输出空行

    测试样例之间输出空行,if(t>0) cout<<endl; 这样出最后一组测试样例之外,其它么每组测试样例之后都会输出一个空行. dp[i]表示以a[i]结尾的最大值,则:dp[i ...

  7. CloudSim样例分析

    自带八个样例描述: cloudsim-2.1.1\examples目录下提供了一些CloudSim样例程序,每个样例模拟的环境如下: (1)CloudSimExample1.Java:创建一个一台主机 ...

  8. 样例20-汽车SHOW

    观看样例点这里 素材下载 1.设置场景大小为400*3002.执行:文件->导入->导入到库,选择需要的汽车图片文件,将其导入到库面板中3.按照同样的方式,在库面板中导入所需的背景音乐文件 ...

  9. java使用xsd校验xml样例

    知识点:XSD文件是指XML结构定义 ( XML Schemas Definition )文件,是DTD的替代品.可以用一个指定的XML Schema来验证某个XML文档,以检查该XML文档是否符合其 ...

随机推荐

  1. 使用SourceTree Push 出现 POST git-receive-pack (chunked) 的解决方法

    在使用SourceTree上传资料的时候,遇到 POST git-receive-pack (chunked) 从 stackoverflow 看到这样一则 This is a bug in Git; ...

  2. 将树莓派变成一个web服务器(2):Nginx+Flask+uWSGI部署全过程

    1)安装Flask,uwsgi,nginx sudo apt-get update sudo apt-get install python-flask #Flask sudo apt-get inst ...

  3. Python实现 zip解压缩到指定目录

    #!/bin/env python #-*- coding:utf-8 -*- import zipfile,os import platform,sys,os from zipfile import ...

  4. Python 字符、整型、列表字典等操作(二)

    在上次课程中简要的讲述了Python的基础常识,现在来详细的学习一下吧! 一.类和对象 面向过程和面向对象 面向过程:C 面向对象:Java.Python等 类和对象的含义: 类,是对事物的抽象,比如 ...

  5. (Problem 6)Sum square difference

    Hence the difference between the sum of the squares of the first ten natural numbers and the square ...

  6. Chapter 6 装饰模式

    修饰模式,是面向对象编程领域中,一种动态地往一个类中添加新的行为的设计模式.就功能而言,修饰模式相比生成子类更为灵活,这样可以给某个对象而不是整个类添加一些功能. 例如:DataOutputStrea ...

  7. RAD Studio 10 自带Demo代码汇总说明

    大家好,好多朋友来信咨询Delphi和C++Builder的移动开发.DataSnap架构等问题,希望能有Demo代码学习.其实Delphi和C++Builder本身自带有很多示例代码,已经覆盖了大部 ...

  8. HTTP 301 跳转和302跳转的区别

    常用的重定向方式有: 301 redirect, 302 redirect 与 meta fresh: 301 redirect::301代表永久性转移(Permanently Moved),301重 ...

  9. kvm libvirt: hostdev passthrough support 解决加密狗冲突问题

    From: "Daniel P. Berrange" <berrange redhat com> To: Guido Günther <agx sigxcpu o ...

  10. UNIX网络编程 卷2:进程间通信

    这篇是计算机类的优质预售推荐>>>><UNIX网络编程 卷2:进程间通信(第2版)> UNIX和网络专家W. Richard Stevens的传世之作 编辑推荐 两 ...