Publish/Subscribe Model——Notification chain——观察者模式
内核中用的很多,整理时间子系统的时候又遇到了notification mechanism,因此做次记录:
参考:1、http://msdn.microsoft.com/en-us/library/ff649664.aspx
2、http://blog.csdn.net/lovelion/article/details/7720232
3、2.6.34
观察者(Observer)模式是对象的行为型模式,又叫做发表-订阅(Publish/Subscribe)模式、模型-视图(Model/View)模式、源-收听者(Source/Listener)模式或从属者(Dependents)模式。
当某件事情发生时,有多个应用需要知道该事件发生并作出相应的反映。
Solution:
Enable listening applications to subscribe to specific messages. Create a mechanism that sends messages to all interested subscribers. The three variations of the Publish/Subscribe pattern you can use to create a mechanism that sends messages to all interested subscribers are List-Based Publish/Subscribe, Broadcast-Based Publish/Subscribe, and Content-Based Publish/Subscribe.
实际上内核中使用的notification 机制就是基于List-Based Publish/Subscribe,下文也着重介绍List-Based Publish/Subscribe,另外两种请阅读第一个参考链接。
List-Based Publish/Subscribe
A List-Based Publish/Subscribe pattern advises you to identify a subject and to maintain a list of subscribers for that subject. When events occur, you have the subject notify each subscriber on the subscription list. When you use this pattern, you identifytwo classes: subjects and observers. Assuming you use a push model update, you add three methods to the subject: Attach(),Detach(),and Notify().You add one method to the observer—Update().
在使用基于链表的 发布——订阅 模式时,我们需要识别出一个subject,而且维护一个对该subject感兴趣的subcriber的链表。当相关事件发生时,我们就通过该subject通知subscription list上的每一个subscriber.当我们使用该模式时,需要鉴别出两个类:subject类和observer类。假设我们使用“push model update"(类似于用链表实现栈),那么我们需要对该subject添加三个方法:Attach(),Detach(),Notifiy(),对于observer则需要添加Update()方法。
To use an observer, all interested observers register with the subject by using the Attach()method. As changes occur to the subject, the subject then calls each registered observer by using the Notify()method.
所有对事件感兴趣的observer(subscriber),通过subject的Attach()方法注册到subscriber list中。当事件发生时,subject通过Notify()方法调用注册在链表上的observer。
Publish/Subscribe的优势:
、降低耦合度:Publisher不知到subcribers的具体信息
、增加安全性:通信机制确保 发布的消息 只被送到注册的 订阅者
Publish/Subcribe的缺点:
、降低性能:如果一个subject有多个直接或间接的observer,通知过程将会占用相当一部分时间
、对程序员提出了高的要求:识别出两个类。
看下内核中的notification mechanism(通知链)是如何实现的:
通知链的运作机制包括两个角色:
、被通知者:对某一事件感兴趣的一方。定义了当事件发生时,相应的处理函数,即回调函数。但需要事先将其注册到通知链中(被通知者注册的动作就是在通知链中增加一项)。
、通知者:事件的通知者。当检测到某个事件时,或者本身产生事件时,通知所有对该事件感兴趣的一方事件发生。它定义了一个通知链,其中保存了每一个被通知者对事件的处理函数(回调函数)。所谓“通知”就是遍历通知链中的每一项,然后调用相应的事件处理函数。 被通知者调用notifier_chain_register函数注册回调函数,该函数按照优先级将回调函数加入到通知链中。
注销回调 函数则使用notifier_chain_unregister函数,即将回调函数从通知链中删除。
通知者调用notifier_call_chain函数通知事件的到达,这个函数会遍历通知链中所有的元素,然后依次调用每一个回调函数(即完成通知动作)。 通知链技术可以概括为:事件的被通知者将事件发生时应该执行的操作通过函数指针方式保存在链表(通知链)中,然后当事件发生时通知者依次执行链表中每一个元素的回调函数完成通知。
通知链节点的数据类型:
struct notifier_block {
int (*notifier_call)(struct notifier_block *, unsigned long, void *);
struct notifier_block *next;
int priority;
};
priority为优先级,对于notification,priority的数值越大意味着优先级越高,在同一链表中会先被调用,当具有相同优先级的节点被插入时,后插入的放在前面(先被调用,
push model)。 从节点中的函数原型可知,返回为int类型,可能的返回值及意义:
NOTIFY_DONE 0X0000 对该事件不感兴趣 NOTIFY_OK 0X0001 成功响应该事件 NOTIFY_STOP_MASK 0X8000 该回调函数返回后停止处理后续notifier block NOTIFY_BAD (NOFITY_STOP_MASK|0X0002) 出错,回调函数返回后停止处理后续notifier block NOTIFY_STOP (NOTIFY_OK|NOFITY_STOP_MASK) 成功响应事件,回调函数返回后停止处理后续notifier block
关于节点中函数原型作用域的unsigned long类型变量的解释:
unsigned long型参数表表示发生的事件类型,因为一个chain可能支持多个事件,此参数用来对事件进行区分。(此处可以看出notification chain与List-Based Pub-Sub
有稍微的不同,因为此处支持多个事件,相当于第一个链接中所谈的topic处理方法)
原子通知链:通知链元素的回调函数在中断上下文运行,不允许阻塞
struct atomic_notifier_head {
spinlock_t lock;
struct notifier_block *head;
};
可阻塞通知链:通知链元素的回调函数在进程上下文运行,允许阻塞:
struct blocking_notifier_head {
struct rw_semaphore rwsem;
struct notifier_block *head;
};
SRCU通知链:可阻塞通知链的一种变体:
struct srcu_notifier_head {
struct mutex mutex;
struct srcu_struct srcu;
struct notifier_block *head;
};
原始通知链:对通知链没有任何限制,所有保护机制都有调用者维护:
struct raw_notifier_head {
struct notifier_block *head;
};
Publish/Subscribe Model——Notification chain——观察者模式的更多相关文章
- (转)RabbitMQ消息队列(四):分发到多Consumer(Publish/Subscribe)
上篇文章中,我们把每个Message都是deliver到某个Consumer.在这篇文章中,我们将会将同一个Message deliver到多个Consumer中.这个模式也被成为 "pub ...
- 【转】Difference between Point-To-Point and Publish/Subscribe JMS Messaging Models
Difference between Point-To-Point and Publish/Subscribe JMS Messaging Models Point-to-Point (PTP) ...
- RabbitMQ消息队列(四):分发到多Consumer(Publish/Subscribe)
上篇文章中,我们把每个Message都是deliver到某个Consumer.在这篇文章中,我们将会将同一个Message deliver到多个Consumer中.这个模式也被成为 "pub ...
- RabbitMQ 分发到多Consumer(Publish/Subscribe)
上篇文章中,我们把每个Message都是deliver到某个Consumer.在这篇文章中,我们将会将同一个Message deliver到多个Consumer中.这个模式也被成为 "pub ...
- RabbitMQ系列教程之三:发布/订阅(Publish/Subscribe)(转载)
RabbitMQ系列教程之三:发布/订阅(Publish/Subscribe) (本教程是使用Net客户端,也就是针对微软技术平台的) 在前一个教程中,我们创建了一个工作队列.工作队列背后的假设是每个 ...
- RabbitMQ消息队列(四):分发到多Consumer(Publish/Subscribe)[转]
上篇文章中,我们把每个Message都是deliver(提供)到某个Consumer.在这篇文章中,我们将会将同一个Message deliver(提供)到多个Consumer中.这个模式也被成为 & ...
- RabbitMQ学习之Publish/Subscribe(3)
上一个教程中,我们创建了一个work queue. 其中的每个task都会被精确的传送到一个worker. 这节,我们将会讲把一个message传送到多个consumers. 这种模式叫做publis ...
- RabbitMQ入门教程(五):扇形交换机发布/订阅(Publish/Subscribe)
原文:RabbitMQ入门教程(五):扇形交换机发布/订阅(Publish/Subscribe) 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. ...
- RabbitMQ的发布订阅模式(Publish/Subscribe)
一.发布/订阅(Publish/Subscribe)模式 发布订阅是我们经常会用到的一种模式,生产者生产消息后,所有订阅者都可以收到.RabbitMQ的发布/订阅模型图如下: 1.该模式下生产者并不是 ...
随机推荐
- BAT-删除文件夹
相关资料:https://www.cnblogs.com/EasonJim/p/6087636.html 可以删除空的文件夹,但是文件夹中有文件,无法删除. @echo off rd "C ...
- Android DrawLayout + ListView 的使用(一)
想做一个APP,设计中有侧边栏这个功能,所以现在开始学习下侧边栏的实现. 在官方的UI空间中已经给出了DrawerLayout这个侧滑的菜单空间. 因为在使用DrawerLayout的时候遇到了些问题 ...
- C# 执行bat批处理文件
private void RunBat(string batPath) { Process pro = new Process(); FileInfo file = new FileInfo(batP ...
- Python3 串口模块移植并使用。
想通过 Python去控制串口模块,直接上层就使用一门语言,这样虽然执行效率低一些,但是开发速度加快 通过 buildroot 先移植 Python-serial 模块 x Symbol: BR2_P ...
- [转]TF-IDF与余弦相似性的应用(一):自动提取关键词
这个标题看上去好像很复杂,其实我要谈的是一个很简单的问题. 有一篇很长的文章,我要用计算机提取它的关键词(Automatic Keyphrase extraction),完全不加以人工干预,请问怎样才 ...
- win10用filezilla server搭建ftp服务器一直无法访问
win10用filezilla server搭建ftp服务器一直无法访问?? 是防火墙导致的,防火墙中允许filezilla server程序的
- IIS 7上部署PHP【后续一】
在之前成功在windows 2008+iis7环境下部署php的基础上,今天把之前的挂Q网站转移到了这个服务器下. 文件拷贝到服务器后,问题继续出现. 首先出现的问题是,Mysql的数据库名称和账户密 ...
- javascript验证键盘keycode
document.onkeyup = function(event){ var event = event || window.event; alert(event.keyCode); }
- SQL Server默认1433端口修改方法
SQL Server默认端口1433端口并不是十分的安全,需要将SQL Server默认端口进行更改,在更改之前,让我们先了解一下什么是1433端口. 什么是1433端口 1433端口,是SQL Se ...
- 关于Unity中如何判断一个动画播放结束
方法一(强力推荐): 在动画结束帧或其他帧处加个动画事件,在播放到这一帧的时候会自动调用这个动画函数 如图,找到对应动画的inspector面板,在里面有个Events下拉条,下拉后在想要的帧的位置添 ...