程序设计模式 —— 观察者模式

 什么是观察者?


 

观察者模式顾名思义就是很多个 类观察主要的类,如果主要的类一旦触发事件,就会通知所有的 观察者类。

如果你不理解观察者模式,我将用一些比较简单的说明来让你理解。

如图

这样说来,所有订报纸的就是观察者,报社就是线程,报社(线程)触发了事件之后会告诉卖报纸(主题)的,叫他去送报纸(发送事件).然后我们观察者就之后了

 

这有什么用?


 

或许有人在学习新知识之前,会问这玩意有什么用,其实我个人认为,观察者设计模式一般用于监听鼠标事件,键盘事件或者其他自定义事件。

因为我们基本上都是写好一个函数,然后等待事件就好了。

那么。代码应该如何实现呢?由于在《head First》 里面已经有Java的实现了,所以我就用C++语言写了一下。

 

框架图


 

首先大致框架图:

大致如上

其中Shell就是 卖报的。所有 卖报的就要实现 Subject 接口(一般而言就一个,当然了你也可以多对多)

Shellp pash ids 都是订报纸的。所有要定报纸的都必须实现 Observe 接口

代码如下:

注意要加上:

1 #include <iostream>
2 #include <vector>  //可以去搜索一下  vector 如何用,如果你熟悉java,可认为它是 ArrayList 类
3 
4 using namespace std;

 

具体代码实现


首先我们先实现 基类

1 /*观察者抽象基类*/
2 class Observer
3 {
4 public:
5     //这里的参数 之所以用string 是为了告诉你可以类传递,甚至也可以传递其他
6     virtual void updata(const char* str) = 0;
7     //注意,在你真实项目中用的时候,应该写一个虚的析构函数
8 
9 };
 1 /*主题 抽象基类*/
 2 class Subject
 3 {
 4 public:
 5 
 6     virtual void eventObserver() = 0;    //事件触发!其他程序来用
 7     virtual void registerObserver(Observer *) = 0;    //注册成为观察者
 8     virtual void removeObserver(Observer *) = 0;    //管理者删除自己
 9 
10 };

 

好了,基本上是这样。那么我们来实现主题:继承subject

 1 /*Shell 继承主题*/
 2 class Shell : public Subject
 3 {
 4 private:
 5     std::vector <Observer*> list;
 6 
 7 public:
 8     //实现方法
 9     inline virtual void eventObserver()
10     {
11         vector<Observer*>::iterator it;
12         int z=0;
13         for (it = list.begin(); it < list.end();it++,z++)    //代送器遍历
14         {
15 
16             (*it)->updata("Hello!");   //向每一个 观察者(监听者) 触发Observe 基类的函数,相当于发送事件
17         }
18     };
19     inline virtual void registerObserver(Observer * ob)//注册成为观察者
20     {
21         list.push_back(ob);//add
22     };
23     inline virtual void removeObserver(Observer * ob)
24     {
25         vector<Observer*>::iterator it;
26         int z=0;
27         for (it = list.begin(); it < list.end();it++,z++)    //代送器遍历
28         {
29             if (&ob == &*it){
30                 //如果地址相同 删除
31                 list.erase(list.begin()+z);
32             }
33         }
34     };
35 };

 

然后我们如概念图,我们节约时间,只实现两个 观察者(相当于就是 事件的监听者)

 1 /*Dis 观察者*/
 2 class Dis : public Observer
 3 {
 4 public:
 5     inline virtual void updata(const char* str)  //这个是Observe基类的方法,这里对其实现,主题(Shell类)会调用这个方法。触发事件
 6     {
 7         cout << "[Dis] DataGet :"<< str << endl;
 8     };
 9 };
10 
11 
12 /*Pash 观察者*/
13 class Pash : public Observer
14 {
15 public:
16     inline virtual ~Pash()
17     {
18 
19     };
20     inline Pash()
21     {
22 
23     };
24     inline virtual void updata(const char* str)   //这个是Observe基类的方法,这里对其实现,主题(Shell类)会调用这个方法。触发事件
25     {
26         cout << "[Pash] DataGet :" << str << endl;
27     };
28 };

上面分别写了两个类,两个类都实现了 updata函数。而且有不同的输出(功能)。

 

下面我们来运行试试?

 1 /*主函数*/
 2 int main()
 3 {
 4     Pash *pash = new Pash();
 5     Dis *dis = new Dis();
 6     Dis *dis2 = new Dis();
 7     Dis *dis3 = new Dis();
 8     Shell shell;
 9     //初始化
10 
11     shell.registerObserver(pash);    //注册
12     shell.registerObserver(dis);
13     shell.registerObserver(dis2);
14     shell.registerObserver(dis3);
15 
16     shell.removeObserver(dis3);    //删除dis3 
17 
18     //触发事件
19     shell.eventObserver();    //这一行不一定非得写在这里,可以在任何需要触发的地方(其他线程)触发
20 
21     //这一行无视~我是用Sublime写的,所以加个这个东西看输出
22     std::cin.get();
23     return 0;
24 }

输出:

 

 

那么到现在为止,我们到底做了什么?


可以看见,这一行:

将 Observe 的子类 Dis(观察者)注册到了 Shell 类(主题)中。并且随时可以等待事件,所有的都一样。

1 shell.registerObserver(dis);


从其他的地方激活事件,然后shell类(主题)就会告诉所有的已经注册了的 Observe 的子类(观察者),触发 “ virtual void updata(const char* str) ”函数;

1  shell.eventObserver();    //这一行不一定非得写在这里,可以在任何需要触发的地方(其他线程)触发

 

怎么触发的呢?

可以去看看Shell类的16行 “ (*it)->updata("Hello!"); ” 对每个 已经注册了的 Observe 子类进行触发。

 

至此,这就是观察者设计模式。

 

最后


但是要注意的一点,不是说有了这个观察者模式,就必须要加进去使用,程序会更棒。

设计模式要与你的程序相互和谐,不能写个 “HelloWorld” 程序都用到了设计模式。

总的一句话,设计模式不是规则,而是你随时可以改变的模式。这也是很多设计模式书籍强调的一点。

 

 

<Thanks>不论对你是否有帮助,还是谢谢您的耐心查看。如有错误之处,还望指教。</Thanks>

posted @ 2016-09-24 15:28  _Suwings  阅读(642)  评论(0编辑  收藏  举报