Observer模式要解决的问题为:建立一个一(Subject)对多(Observer)的依赖关系,并且做到当“一”变化的时候,依赖这个“一”的多也能够同步改变。最常见的一个例子就是:对同一组数据进行统计分析时候,我们希望

能够提供多种形式的表示 (例如以表格进行统计显示、 柱状图统计显示、 百分比统计显示等) 。这些表示都依赖于同一组数据,我们当然需要当数据改变的时候,所有的统计的显示都能够同时改变。Observer模式就是解决了这一个问题。

 /////////////Subject.h/////////////////////////////////////////////////////////////
#pragma once
#include <string>
#include <list>
using namespace std;
class Observer ; typedef string State ;
class Subject
{
public:
virtual ~Subject();
virtual void Attach(Observer* obv);
virtual void Detach(Observer* obv);
virtual void Notify();
virtual void SetState(const State& st) = ;
virtual State GetState() = ;
protected:
Subject(); private:
list<Observer*>* _Obvs ;
}; class ConcreteSubject : public Subject
{
public:
~ConcreteSubject();
ConcreteSubject();
State GetState();
void SetState(const State& st) ;
protected:
private:
State _st ;
};
 //////////////////Observer.h////////////////////////////////////////////////////////
#pragma once
#include <string>
using namespace std;
typedef string State ;
class Subject ; class Observer
{
public:
virtual ~Observer();
virtual void Update(Subject* sub) = ;
virtual void PrintInfo() = ;
State _st ;
protected:
Observer(); private:
}; class ConcreteObserverA : public Observer
{
public:
~ConcreteObserverA();
ConcreteObserverA(Subject* sub);
virtual Subject* GetSubject();
void Update(Subject* sub);
void PrintInfo();
protected:
private:
Subject* _sub ;
}; class ConcreteObserverB : public Observer
{
public:
~ConcreteObserverB();
ConcreteObserverB(Subject* sub);
virtual Subject* GetSubject();
void Update(Subject* sub);
void PrintInfo();
protected:
private:
Subject* _sub ;
};
 ////////////Subject.cpp//////////////////////////////////////////////////////////////
#include "Subject.h"
#include "Observer.h"
#include <list>
typedef string State ; Subject::Subject()
{
_Obvs = new list<Observer*>();//在模板的使用之前一定要new
}
Subject::~Subject()
{ } void Subject::Attach(Observer* obv)
{
_Obvs->push_front(obv);
}
void Subject::Detach(Observer* obv)
{
if (obv != NULL)
{
_Obvs->remove(obv);
}
}
void Subject::Notify()
{
list<Observer*>::iterator it = _Obvs->begin();
for (;it != _Obvs->end(); it++)
{
(*it)->Update(this);
}
} ConcreteSubject::~ConcreteSubject()
{ }
ConcreteSubject::ConcreteSubject()
{
_st = '\0';
}
State ConcreteSubject::GetState()
{
return _st ;
}
void ConcreteSubject::SetState(const State& st)
{
_st = st ;
}
 ////////////Observer.cpp//////////////////////////////////////////////////////////////
#include "Observer.h"
#include "Subject.h"
#include <iostream>
using namespace std ;
Observer::Observer()
{
_st = '\0';
}
Observer::~Observer()
{ } ConcreteObserverA::ConcreteObserverA(Subject* sub)
{
_sub = sub ;
_sub->Attach(this);
}
ConcreteObserverA::~ConcreteObserverA()
{
_sub->Detach(this);
if (_sub != NULL)
{
delete _sub ;
}
} Subject* ConcreteObserverA::GetSubject()
{
return _sub ;
}
void ConcreteObserverA::PrintInfo()
{
cout<<"观察者A的到的通知:"<<_sub->GetState()<<endl;
} void ConcreteObserverA::Update(Subject* sub)
{
_st = sub->GetState();
PrintInfo();
} ConcreteObserverB::ConcreteObserverB(Subject* sub)
{
_sub = sub ;
_sub->Attach(this);
}
ConcreteObserverB::~ConcreteObserverB()
{
_sub->Detach(this);
if (_sub != NULL)
{
delete _sub;
}
}
Subject* ConcreteObserverB::GetSubject()
{
return _sub ;
} void ConcreteObserverB::PrintInfo()
{
cout<<"观察者B的到的通知:"<<_sub->GetState()<<endl;
}
void ConcreteObserverB::Update(Subject* sub)
{
_st = sub->GetState();
PrintInfo();
}
 /////////////main.cpp/////////////////////////////////////////////////////////////
#include "Subject.h"
#include "Observer.h"
#include <iostream>
using namespace std; int main()
{
Subject* sub = new ConcreteSubject();
Observer* o1 = new ConcreteObserverA(sub);
Observer* o2 = new ConcreteObserverB(sub); sub->SetState("你获得一封邮件!");
sub->Notify();
sub->SetState("你的评论有新的回复!");
sub->Notify(); getchar();
return ;
}

Observer 模式的更多相关文章

  1. C# 关于委托和事件的妙文:通过一个例子详细介绍委托和事件的作用;Observer模式简介

    委托和事件在 .Net Framework中的应用非常广泛,然而,较好地理解委托和事件对很多接触C#时间不长的人来说并不容易.它们就像是一道槛儿,过了这个槛的人,觉得真是太容易了,而没有过去的人每次见 ...

  2. 【行为型】Observer模式

    观察者模式意图解决一对多的依赖关系情形中,当被依赖对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新的问题.举个很简单的例子说,假如一个游戏中的角色的某个属性状态发生了变化,此处不妨假设为 ...

  3. Java 实现观察者(Observer)模式

    1. Java自带的实现 类图 /** * 观察目标 继承自 java.util.Observable * @author stone * */ public class UpdateObservab ...

  4. Behavioral模式之Observer模式

    1.意图 定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,全部依赖于它的对象都得到通知并被自己主动更新. 2.别名 依赖(dependents).公布-订阅(Publish-Subscr ...

  5. 3)Javascript设计模式:Observer模式

    Observer模式 var Observer = (function() { var instance = null; function Observe() { this.events = {} } ...

  6. Java设计模式(20)观察者模式(Observer模式)

    Java深入到一定程度,就不可避免的碰到设计模式(design pattern)这一概念,了解设计模式,将使自己对java中的接口或抽象类应用有更深的理解.设计模式在java的中型系统中应用广泛,遵循 ...

  7. Observer模式实践

    Observer 模式在实践中的应用场景: 为 Point 类设计一个数据绑定机制,当其坐标 x 或 y 被更改时,可以通知外界其更改的过程.将更改过程打印在控制台上.考虑使用松耦合设计. 代码: # ...

  8. 设计模式之——Observer模式

    Observer模式又叫做观察者模式,当观察对象状态发生变化的时候,就会通知给观察者.这种模式适用于根据对象状态进行响应的场景! 实例程序是一个输出数字的程序. 观察者Observer类用于每500m ...

  9. Observer模式(观察者设计模式)

    Observer 设计模式? 在Observer模式中,当观察对象的状态发生变化时,会通知给观察者.Observer模式适用于根据对象状态进行相应处理的场景. Observer 并非主动观察,而是被动 ...

随机推荐

  1. 如何在 Linux 终端下创建新的文件系统/分区

    在 Linux 中创建分区或新的文件系统通常意味着一件事:安装 Gnome Parted 分区编辑器(GParted).对于大多数 Linux 用户而言,这是唯一的办法.不过,你是否考虑过在终端创建这 ...

  2. 001-python基础

    1.Python是什么样的语言? 1.1语言的分类: 编译型 解释型 混合型 编译型: 优点:编译器一般会有预编译的过程对代码进行优化. 因为编译只做一次,运行时不需要编译,所以编译型 语言的程序执行 ...

  3. (7/18)重学Standford_iOS7开发_视图、绘制、手势识别_课程笔记

    第七课: 1.View 一般来说,视图是一个构造块,代表屏幕上一块矩形区域,定义了一个坐标空间,并在其中绘制及添加触控事件等. ①视图的层级关系 一个视图只能有一个父视图,可以有多个子视图 - ( - ...

  4. tomcat报错:This is very likely to create a memory leak问题解决

    tomcat memory leak解决方案 这种问题在开发中经常会碰到的,看看前辈的总结经验 Tomcat内存溢出的原因  在生产环境中tomcat内存设置不好很容易出现内存溢出.造成内存溢出是不一 ...

  5. 通过rest接口获取自增id (twitter snowflake算法)

    1.  算法介绍 参考 http://www.lanindex.com/twitter-snowflake%EF%BC%8C64%E4%BD%8D%E8%87%AA%E5%A2%9Eid%E7%AE% ...

  6. 在Windows下使用MinGW静态编译Assimp

    使用MinGW静态编译Assimp 到了5月份了,没有写一篇日志,于是自己从知识库里面拿出一篇文章充数吧.这次将要解说怎样在Windows下使用MinGW静态编译Assimp. Assimp是眼下比較 ...

  7. java中使用队列:java.util.Queue

    在java5中新添加了java.util.Queue接口,用以支持队列的常见操作.该接口扩展了java.util.Collection接口.Queue使用时要尽量避免Collection的add()和 ...

  8. classpath多个包添加

    转的: 把某目录下的所有子目录和子目录下面的下层和更下层目录也添加到CLASSPATH里面,更具需求写了两句,发上来分享,备忘. 在Linux下可以通过这样的方法把/opt/.../lib目录下的所有 ...

  9. WIN32 根据程序名(映像名称)终止外部程序

    场景: 1.有时候需要调用外部程序,但是外部程序有可能崩溃挂起,这样这个进程就没法结束,所以再在下次调用前需要先结束之前调用的. 2.没考虑到权限问题,应该是只能终止当前用户启动的进程. #inclu ...

  10. 看懂下面C++代码才说你理解了C++多态虚函数!

    #include <iostream> using namespace std ; class Father { private :  virtual void Say()  //只有添加 ...