usr/include/dispatch - dispatch_source
博文一部分摘自:Parse分析,以下简称博文1(LeanCloud工程师针对Parse使用GCD的分析)
博文一部分摘自:GCD入门,以下简称博文2
建议先了解一下:BSD基础知识
在Dispatch/source.h
中是这样描述Dispatch Source####
The dispatch framework provides a suite of Interfaces for monitoring low-level system objects (file descriptions , Mach Ports , signals , VFS nodes , etc.)
for activity and automatically submitting event handler blocks to dispatch queues when such activity occurs.
The suite of interfaces is known as the Dispatch Source API
abstract:
Dispatch Source are used to automatically submit event handler blocks to dispatch queues in response to external events.
简单来说####
Dispatch Source 是用来监听某些类型事件的对象,当这些事件发生时,Dispatch Source 自动把事件block放入到Dispatch queue的执行例程中。
Dispatch Source 监听的底层事件####
下面是GCD 10.6.0 版本支持的事件:
- Mach port send right state changes.
- Mach port receive right state changes.
- External process state changes.
- File descriptor ready for read.
- File descriptor ready for write.
- File system node event.
- POSIX signal.
- Custom timer.
- Custom event.
这些事件支持所有kqueue支持的事件、mach端口、内建计时器支持(这样我们就不用使用timeout参数来创建自己的计时器)和用户事件。
如果对这些知识点不了解,可以看我对知识点的整理 - 知识点资料 3 ~ 6
一、 Custom event
博文1中说Dispatch Source
比使用dispatch_async
的优势在于利用合并
。
合并:在任一thread调用它的一个函数dispatch_source_merge_data
后,会执行Dispatch Source
事先定义好的event handler(博主1认为是block)。
博文以总结说,联结的这个过程就称为 - Custom event(用户事件)
。是dispatch source支持处理的一种事件。简单说,这种事件是由你调用dispatch_source_merge_data
函数来向自己发送信号。
1.1 Dispatch Source 的使用步骤 详情介绍点 - 这里###
- 创建一个 Dispatch Source
- 创建 Dispatch Source 的事件句柄
- 处理 Dispatch Source 的resume() 和 suspend() 操作。
- 向 Dispatch Source 发送事件。
1.1.1 创建一个 Dispatch Source
dispatch_source_create(<#dispatch_source_type_t type#>,
<#uintptr_t handle#>,
<#unsigned long mask#>,
<#dispatch_queue_t queue#>);
/** 参数1
* dispatch_source_type_t type
* 需要监听的事件类型
*/
/** 参数2
* uintptr_t handle
* 理解为 id、索引或句柄。假如要监听process,需要传入process的pId.
*/
/** 参数3
* unsigned long mask
* 根据参数2,理解为description、provide或detail description,让source知道具体要监听什么。
*/
/** 参数4
* dispatch_queue_t queue
* 当监听事件发生,将event_handle_block添加到这个queue来执行。
*/
dispatch_source_type_t type
DISPATCH_SOURCE_TYPE_DATA_ADD; //*> custom event, 自己定义事件,自己触发事件
DISPATCH_SOURCE_TYPE_DATA_OR; //*> custom event, 自己定义事件,自己触发事件
DISPATCH_SOURCE_TYPE_MACH_SEND; //*> Mach 相关‘发送’事件
DISPATCH_SOURCE_TYPE_MACH_RECV; //*> Mach 相关‘接收’事件
DISPATCH_SOURCE_TYPE_READ; //*> I/O 操作,对文件、socket操作的‘读’响应
DISPATCH_SOURCE_TYPE_WRITE; //*> I/O 操作,对文件、socket操纵的‘写’响应
DISPATCH_SOURCE_TYPE_VNODE; //*> 文件状态监听,文件被删除、移动、重命名
DISPATCH_SOURCE_TYPE_PROC; //*> 进程监听,如,进程的退出、创建1或多个进程、进程收到Unix信号
DISPATCH_SOURCE_TYPE_TIMER; //*> 定时响应
DISPATCH_SOURCE_TYPE_SIGNAL; //*> 接收Unix信号时的响应
创建一个dispatch_source_t source
dispatch_source_t customSourceMonitor = dispatch_source_create(DISPATCH_SOURCE_TYPE_DATA_ADD, 0, 0, dispatch_get_main_queue());
1.1.2 创建Dispatch_source_t 的event_handler_block###
/**
* 如果dispatch_source的监听类型:
* 1.DISPATCH_SOURCE_TYPE_DATA_ADD
* 2.DISPATCH_SOURCE_TYPE_DATA_OR
* 这是Custom Event,自定定义Handler_Block,自己去触发Handler_Block
*/
dispatch_source_set_event_handler(customSourceMonitor, ^{
// TODO
});
1.1.3 处理 Dispatch_source_t 的suspend 、 resume 操作。
初始化创建的Dispatch Source 默认是suspend。需要resume
dispatch_resume(customSourceMonitor);
dispatch_suspend(customSourceMonitor);
1.14 向 Dispatch_Source_t 发送事件
(下文2.1 详解)
resume Dispatch_Source_t 之后,可以通过 dispatch_source_merge_data
向 Dispatch_Source 发送event。
- 当dispatch_source_merge_data触发了handler_event 之后,dispatch_source_get_data(Dispatch_source_t source)就会自动清空成‘0’。
1.15 Custom Event 演示代码
dispatch_source_t customSourceMonitor = dispatch_source_create(DISPATCH_SOURCE_TYPE_DATA_ADD, 0, 0, dispatch_get_main_queue());
dispatch_resume(customSourceMonitor);
__block int sum = 0;
dispatch_source_set_event_handler(customSourceMonitor, ^{
// ?TODO
sum += dispatch_source_get_data(customSourceMonitor);
NSLog(@"- %d",sum);
});
//4950
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
for (NSUInteger i = 0; i < 1000; i ++) {
dispatch_async(queue, ^{
dispatch_source_merge_data(customSourceMonitor, i);
usleep(20000); // 注释效果不同
});
}
二、如何使用Dispatch Source 达到 NSOperationQueue 的功能
2.1 Dispatch Source 如何通过merge的方式确保在高负载下正常工作
1.14 中通过调用dispatch_source_merge_data
后所触发dispatch_source_set_event_handler
的根据merge特性调用的频率会大大减少,有时候只调用一次handler。
为什么Dispatch Source 能通过 merge 方式确保在高负载下正常运行呢?
在同一时间,只有一个处理block的实例被分配,如果这个handler_block还没有执行完毕,另一个事件就发生了,事件会以指定的方式 (ADD 或 OR) 进行积累。
Dispatch Source 能通过merge事件确保在高负载下正常工作。当handler_event被执行的时,可以通过dispatch_source_get_data() 来获取计算后累加的数据。然后这个dispatch_source_get_data() 会被重置为‘0’。
2.2 Dispatch Source 与 Dispatch Queue 两者在线程执行上的关系
答: 没有关系
usr/include/dispatch - dispatch_source的更多相关文章
- /usr/include/sys/types.h:62: error: conflicting types for ‘dev_t’
/usr/include/sys/types.h:62: error: conflicting types for ‘dev_t’/usr/include/linux/types.h:13: erro ...
- 解决Cannot find MySQL header files under /usr/include/mysql的错误
按照下面的步骤能成功,亲测.转帖,做笔记 编译php-5.5-6的mysql支持,出现Cannot find MySQL header files under /usr/include/mysql. ...
- 编译安装php Cannot find MySQL header files under /usr/include/mysql.
编译php-5.5-6的mysql支持,出现Cannot find MySQL header files under /usr/include/mysql. Note that the MySQL c ...
- error: /usr/include/objc/objc-class.h: No such file or directory
When i use the example of ShareKit package,i have come across this error:"error: /usr/include/o ...
- error: /usr/include/stdio.h: Permission denied 的一种情况分析
error: /usr/include/stdio.h: Permission denied 的一种情况分析 代码: #include <stdio.h> int main(){ prin ...
- Mac下一个/usr/include失踪
Mac升级到Yosemite后,突然发现vim的YouCompleteMe代码提示所以空头支票成员,排查了一下,原本/usr/include目录中缺少.所有的C/C++头文件不见了. .. 第一次发现 ...
- ubuntu没有/usr/include/sys目录
实际上不是没有sys目录,只是系统给换路径了 32位系统:/usr/incude/i386-linux-gnu/sys 64位系统:/usr/include/x86_64-linux-gnu/sys/ ...
- [C++]Linux之头文件sys/types.h[/usr/include/sys]
1.查找<sys/types.h>文件 一般地,Linux的C头文件<sys/types.h>路径在如题的途径:/usr/include/sys下,然而博主[Linux For ...
- usr/include/c++/6.4.1/bits/stl_relops.:67: Parse error at "std"
问题描述: 1.编译某qt工程的32位架构二进制包时,出现了上面错误,具体错误信息如下 qmake-qt5 -o ProductLicense/Makefile ProductLicense/Prod ...
随机推荐
- ORA-00600: internal error code, arguments: [17281], [1001], [0x1FF863EE8], [], [], [], [], []
我们生产服务器中的一个数据库发出监控告警日志的邮件,内容如下所示,在31号09:11分出现了大名鼎鼎的ORA-00600错误. Dear All: The Instance xxx' alert lo ...
- Redhat Server 5.7 安装配置PHP
PHP的简介 PHP于1994年由Rasmus Lerdorf创建,刚刚开始是Rasmus Lerdorf 为了要维护个人网页而制作的一个简单的用Perl语言编写的程序.这些工具程序用来显示 Rasm ...
- .NET应用架构设计—适当使用活动记录模式代替领域模型模式
阅读目录: 1.背景介绍 2.简单介绍领域模型模式.活动记录模式 3.活动记录模式的简单示例及要点 4.总结 1.背景介绍 对软件开发方法论有兴趣的博友应该发现最近“领域驱动设计”慢慢的被人发现被人实 ...
- PostgreSQL-数据目录与pg_ctl
# tail /etc/profile PATH="$PATH":/usr/lib/postgresql/9.2/bin/ export PATH export PGDATA=/v ...
- 开发人员必读openstack网络基础
云计算中的网络非常复杂,需要对网络的基础理论有一定的认识和了解,转载网上针对openstack中涉及到网络概念的文章 开发人员必读openstack网络基础1:什么是L2.L3 开发人员必读opens ...
- [WPF系列]-Adorner
简介 通常我们想对现有的控件,做些修饰时我们就会想到一个装饰模式.WPF中也提供了这样的实现思路:通过将Adorner添加到AdornerLayer中来实现装饰现有控件的效果.如图示: 本来T ...
- 理解 OpenStack 高可用(HA)(3):Neutron 分布式虚拟路由(Neutron Distributed Virtual Routing)
本系列会分析OpenStack 的高可用性(HA)概念和解决方案: (1)OpenStack 高可用方案概述 (2)Neutron L3 Agent HA - VRRP (虚拟路由冗余协议) (3)N ...
- MVC架构学习之Smarty学习——病来而蔫
前两天是五一小长假,而每次假期都想着如何如何刻苦一番,往往是自作多情.. 当然这次是有小病在身,多个借口吧. 一有病就蔫的不行...要锻炼了啊,脚估计也差不多了,游泳试试吧这周. 这次学习Smarty ...
- 数据库SQL基本操作
1.查询表结构 desc 表名 2.显示当前连接用户 show user 3.查看系统拥有哪些用户 select * from all_users; 4.查询当前用户下所有对象 select * fr ...
- 三维等值面提取算法(Dual Contouring)
上一篇介绍了Marching Cubes算法,Marching Cubes算法是三维重建算法中的经典算法,算法主要思想是检测与等值面相交的体素单元并计算交点的坐标,然后对不同的相交情况利用查找表在体素 ...