iOS dispatch_source_t的理解
Dispatch Source是GCD中的一个基本类型,从字面意思可称为调度源,它的作用是当有一些特定的较底层的系统事件发生时,调度源会捕捉到这些事件,然后可以做其他的逻辑处理,调度源有多种类型,分别监听对应类型的系统事件。我们来看看它都有哪些类型:
- Timer Dispatch Source:定时调度源。
- Signal Dispatch Source:监听UNIX信号调度源,比如监听代表挂起指令的SIGSTOP信号。
- Descriptor Dispatch Source:监听文件相关操作和Socket相关操作的调度源。
- Process Dispatch Source:监听进程相关状态的调度源。
- Mach port Dispatch Source:监听Mach相关事件的调度源。
- Custom Dispatch Source:监听自定义事件的调度源。
用通俗一点的话说就是用GCD的函数指定一个希望监听的系统事件类型,再指定一个捕获到事件后进行逻辑处理的闭包或者函数作为回调函数,然后再指定一个该回调函数执行的Dispatch Queue即可,当监听到指定的系统事件发生时会调用回调函数,将该回调函数作为一个任务放入指定的队列中执行。也就是说当监听到系统事件后就会触发一个任务,并自动将其加入队列执行,这里与之前手动添加任务的模式不同,一旦将Diaptach Source与Dispatch Queue关联后,只要监听到系统事件,Dispatch Source就会自动将任务(回调函数)添加到关联的队列中。有些时候回调函数执行的时间较长,在这段时间内Dispatch Source又监听到多个系统事件,理论上就会形成事件积压,但好在Dispatch Source有很好的机制解决这个问题,当有多个事件积压时会根据事件类型,将它们进行关联和结合,形成一个新的事件。
监听事件类型
Dispatch Source一共可以监听六类事件,分为11个类型,我们来看看都是什么:
DISPATCH_SOURCE_TYPE_DATA_ADD
:属于自定义事件,可以通过dispatch_source_get_data
函数获取事件变量数据,在我们自定义的方法中可以调用dispatch_source_merge_data
函数向Dispatch Source设置数据,下文中会有详细的演示。DISPATCH_SOURCE_TYPE_DATA_OR
:属于自定义事件,用法同上面的类型一样。DISPATCH_SOURCE_TYPE_MACH_SEND
:Mach端口发送事件。DISPATCH_SOURCE_TYPE_MACH_RECV
:Mach端口接收事件。DISPATCH_SOURCE_TYPE_PROC
:与进程相关的事件。DISPATCH_SOURCE_TYPE_READ
:读文件事件。DISPATCH_SOURCE_TYPE_WRITE
:写文件事件。DISPATCH_SOURCE_TYPE_VNODE
:文件属性更改事件。DISPATCH_SOURCE_TYPE_SIGNAL
:接收信号事件。DISPATCH_SOURCE_TYPE_TIMER
:定时器事件。DISPATCH_SOURCE_TYPE_MEMORYPRESSURE
:内存压力事件。
创建Dispatch Source
我们可以使用dispatch_source_create
函数创建Dispatch Source,该函数有四个参数:
type
:第一个参数用于标识Dispatch Source要监听的事件类型,共有11个类型。handle
:第二个参数是取决于要监听的事件类型,比如如果是监听Mach端口相关的事件,那么该参数就是mach_port_t
类型的Mach端口号,如果是监听事件变量数据类型的事件那么该参数就不需要,设置为0就可以了。mask
:第三个参数同样取决于要监听的事件类型,比如如果是监听文件属性更改的事件,那么该参数就标识文件的哪个属性,比如DISPATCH_VNODE_RENAME
。queue
:第四个参数设置回调函数所在的队列。
NSTimeInterval period = 0.1; //设置时间间隔
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);//dispatch_source_t _timer 类型的
设置事件处理器
前文中提到过,当Dispatch Source监听到事件时会调用指定的回调函数或闭包,该回调函数或闭包就是Dispatch Source的事件处理器。我们可以使用dispatch_source_set_event_handler
或dispatch_source_set_event_handler_f
函数给创建好的Dispatch Source设置处理器,前者是设置闭包形式的处理器,后者是设置函数形式的处理器:
dispatch_source_set_event_handler(dispatchSource, {
print("Dispatch Source 事件处理器...")})// 根据闭包尾随的特性,还可以有下面的写法dispatch_source_set_event_handler(dispatchSource) {
print("Dispatch Source 事件处理器...")
}
例子:
__block int timeout=300; //倒计时时间
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0,queue);
dispatch_source_set_timer(_timer,dispatch_walltime(NULL, 0),1.0*NSEC_PER_SEC, 0); //没秒执行
dispatch_source_set_event_handler(_timer, ^{
if(timeout<=0){ //倒计时结束,关闭
dispatch_source_cancel(_timer);
dispatch_release(_timer);
dispatch_async(dispatch_get_main_queue(), ^{
//设置界面的按钮显示 根据自己需求设置
。。。。。。。。
});
}else{
int minutes = timeout / 60;
int seconds = timeout % 60;
NSString *strTime = [NSString stringWithFormat:@"%d分%.2d秒后重新获取验证码",minutes, seconds];
dispatch_async(dispatch_get_main_queue(), ^{
//设置界面的按钮显示 根据自己需求设置
。。。。。。。。
});
timeout--; }
});
dispatch_resume(_timer);
既然是事件处理器,那么肯定需要获取一些Dispatch Source的信息,GCD提供了三个在处理器中获取Dispatch Source相关信息的函数,比如handle
、mask
。而且针对不同类型的Dispatch Source,这三个函数返回数据的值和类型都会不一样,下面来看看这三个函数:
dispatch_source_get_handle
:这个函数用于获取在创建Dispatch Source时设置的第二个参数handle
。- 如果是读写文件的Dispatch Source,返回的就是描述符。
- 如果是信号类型的Dispatch Source,返回的是
int
类型的信号数。 - 如果是进程类型的Dispatch Source,返回的是
pid_t
类型的进程id。 - 如果是Mach端口类型的Dispatch Source,返回的是
mach_port_t
类型的Mach端口。
dispatch_source_get_data
:该函数用于获取Dispatch Source监听到事件的相关数据。- 如果是读文件类型的Dispatch Source,返回的是读到文件内容的字节数。
- 如果是写文件类型的Dispatch Source,返回的是文件是否可写的标识符,正数表示可写,负数表示不可写。
- 如果是监听文件属性更改类型的Dispatch Source,返回的是监听到的有更改的文件属性,用常量表示,比如
DISPATCH_VNODE_RENAME
等。 - 如果是进程类型的Dispatch Source,返回监听到的进程状态,用常量表示,比如
DISPATCH_PROC_EXIT
等。 - 如果是Mach端口类型的Dispatch Source,返回Mach端口的状态,用常量表示,比如
DISPATCH_MACH_SEND_DEAD
等。 - 如果是自定义事件类型的Dispatch Source,返回使用
dispatch_source_merge_data
函数设置的数据。
dispatch_source_get_mask
:该函数用于获取在创建Dispatch Source时设置的第三个参数mask
。在进程类型,文件属性更改类型,Mach端口类型的Dispatch Source下该函数返回的结果与dispatch_source_get_data
一样。
注册Cancellation Handler
Cancellation Handler就是当Dispatch Source被释放时用来处理一些后续事情,比如关闭文件描述符或者释放Mach端口等。我们可以使用dispatch_source_set_cancel_handler
函数或者dispatch_source_set_cancel_handler_f
函数给Dispatch Source注册Cancellation Handler:
dispatch_source_set_cancel_handler(dispatchSource) { print("进行善后处理...") }
该函数有两个参数,第一个参数是目标Dispatch Source,第二个参数就是要进行善后处理的闭包或者函数。
更改Dispatch Source的目标队列
在上文中,我们说过可以使用dispatch_source_create
函数创建Dispatch Source,并且在创建时会指定回调函数执行的队列,那么如果事后想更改队列,比如说想更改队列的优先级,这时我们可以使用dispatch_set_target_queue
函数实现:swift
let dispatchQueueDefaultPriority = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0) let dispatchSource = dispatch_source_create(DISPATCH_SOURCE_TYPE_DATA_ADD, 0, 0, dispatchQueueDefaultPriority) let dispatchQueueLowPriority = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0) dispatch_set_target_queue(dispatchSource, dispatchQueueLowPriority)
这里需要注意的是,如果在更改目标队列时,Dispatch Source已经监听到相关事件,并且回调函数已经在之前的队列中执行了,那么会一直在旧的队列中执行完成,不会转移到新的队列中去。
暂停恢复Dispatch Source
暂停和恢复Dispatch Source与Dispatch Queue一样,都适用dispatch_suspend
和dispatch_resume
函数。这里需要注意的是刚创建好的Dispatch Source是处于暂停状态的,所以使用时需要用dispatch_resume
函数将其启动。
废除Dispatch Source
如果我们不再需要使用某个Dispatch Source时,可以使用dispatch_source_cancel
函数废除,该函数只有一个参数,那就是目标Dispatch Source。
Dispatch Source实践
说了这么多,这一节来看看Dispatch Source到底怎么用。
用Dispatch Source监听定时器
Dispatch Source能监听的事件中有一个类型就是定时器,我们来看看如何实现:swift
class TestDispatchSource {
func launch() {
let dispatchQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0)
let timer = createTimerDispatchSource(dispatch_time(DISPATCH_TIME_NOW, 0), interval: NSEC_PER_SEC * 5, leeway: 0, queue: dispatchQueue) {
print("处理定时任务,该任务每5秒执行一次...") } dispatch_resume(timer) sleep(30) }
func createTimerDispatchSource(startTime: dispatch_time_t, interval: UInt64, leeway: UInt64, queue: dispatch_queue_t, handler: dispatch_block_t) -> dispatch_source_t {
let timerDispatchSource = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue)
dispatch_source_set_timer(timerDispatchSource, startTime, interval, leeway)
dispatch_source_set_event_handler(timerDispatchSource, handler)
return timerDispatchSource }}
上面的代码示例中一个新的函数dispatch_source_set_timer
,该函数的作用就是给监听事件类型为DISPATCH_SOURCE_TYPE_TIMER
的Dispatch Source设置相关属性,该函数有四个参数:
source
:该参数为目标Dispatch Source,类型为dispatch_source_t
.start
:该参数为定时器的起始时间,类型为dispatch_time_t
。interval
:该参数为定时器的间隔时间,类型为UInt64
,间隔时间的单位是纳秒。leeway
:该参数为间隔时间的精度,类型为UInt64
,时间单位也是纳秒。
用Dispatch Source监听自定义事件
Dispatch Source能监听的事件中有一个类型是自定义事件,下面我们来看看如何使用:
class TestDispatchSource {
func launch() {
var totalProcess = 0
let dispatchSource = dispatch_source_create(DISPATCH_SOURCE_TYPE_DATA_ADD, 0, 0, dispatch_get_main_queue()) dispatch_source_set_event_handler(dispatchSource) {
let process = dispatch_source_get_data(dispatchSource)
totalProcess += Int(process)
print("这里可以在主线程更新UI,显示进度条...进度为/(totalProcess)%")
}
dispatch_resume(dispatchSource)
generateCustomEvent(dispatchSource)
} func generateCustomEvent(dispatchSource: dispatch_source_t) {
let queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0)
for index in 0...100 {
dispatch_sync(queue) {
print("模拟自定义事件...进度为/(index)%")
dispatch_source_merge_data(dispatchSource, 1)
sleep(2)
}
}
}
}
我们来看看generateCustomEvent(dispatchSource: dispatch_source_t)
方法,该方法的作用的是模拟自定义事件,首先创建一个全局并发队列,然后循环让其执行任务,在执行的任务里调用dispatch_source_merge_data
函数,就可以触发监听类型为DISPATCH_SOURCE_TYPE_DATA_ADD
或者DISPATCH_SOURCE_TYPE_DATA_OR
的Dispatch Source。该函数有两个参数,第一个参数是目标Dispatch Source,第二个参数的类型是无符号长整型,用于向目标Dispatch Source中的对应变量追加指定的数。
我们再来看看如何监听自定义时间,首先创建类型为DISPATCH_SOURCE_TYPE_DATA_ADD
的Dispatch Source,然后设置回调闭包,在闭包中使用dispatch_source_get_data
获取追加的变量值,该函数只有一个参数,就是目标Dispatch Source,这里需要注意的是通过dispatch_source_get_data
函数获取的变量值并不是累加值,而是每次调用dispatch_source_merge_data
函数时设置的值,所以在上面的示例中用totalProcess
变量累加每次获取到的值。
上面的示例可以用来模拟后台进行下载,根据下载的数据量使用dispatch_source_merge_data
函数给目标Dispatch Source设置相应的变量值,然后在主线程中监听到Dispatch Source的自定义事件,通过dispatch_source_get_data
函数获取到变量,用于更新显示进度条的UI。
原文链接:http://www.th7.cn/Program/IOS/201605/849625.shtml
参考:http://blog.csdn.net/kut00/article/details/8845351
iOS dispatch_source_t的理解的更多相关文章
- iOS开发SDWebImageOptions理解
iOS开发SDWebImageOptions理解 原文 http://www.cnblogs.com/WJJ-Dream/p/5816750.html typedef NS_OPTIONS(NSUIn ...
- iOS刨根问底-深入理解RunLoop
开源的RunloopRef 通常所说的RunLoop指的是NSRunloop或者CFRunloopRef,CFRunloopRef是纯C的函数,而NSRunloop仅仅是CFRunloopRef的OC ...
- 在真机调试 iOS 应用:理解 Certificates, Identifiers & Profiles
No matching provisioning profiles found. No matching code signing identity found. Your account alrea ...
- iOS UIStackView的理解
iOS9 提供的UIStackview简化了布局操作,它有些像Android中的liner layout.以前不用UIStackview也是可以做出类似的效果的,不过要添加许多的约束,看得头都大了,使 ...
- iOS代码签名理解
前言 做了几年iOS app coder了,对于证书的生成.使用流程烂熟于心,然而对于这套机制的原理却一直不甚理解.近来由于工作需要仔细研究了一下,特将自己的学习经验记录于此,以供大家学习指正. 问题 ...
- IOS UIView 02- 深入理解 Scroll Views
注:本人是翻译过来,并且加上本人的一点见解. 前言 可能你很难相信 UIScrollView 和一个标准的 UIView 差异并不大,scroll view 确实会多出一些方法,但这些方法只是和 UI ...
- iOS 的一点理解(一) 代理delegate
做了一年的iOS,想记录自己对知识点的一点理解. 第一篇,想记录一下iOS中delegate(委托,也有人称作代理)的理解吧. 故名思议,delegate就是代理的含义, 一件事情自己不方便做,然后交 ...
- IOS GCD 的理解
GCD (Grand Central Dispatch) 是Apple公司开发的一种技术,它旨在优化多核环境中的并发操作并取代传统多线程的编程模式. 在Mac OS X 10.6和IOS 4.0之后开 ...
- 0112.1——iOS开发之理解iOS中的MVC设计模式
模型-视图-控制器(Model-View-Controller,MVC)是Xerox PARC在20世纪80年代为编程语言Smalltalk-80发明的一种软件设计模式,至今已广泛应用于用户交互应用程 ...
随机推荐
- BaseServlet方法分发
BaseServlet.java package org.guangsoft.controller; import java.io.IOException; import java.lang.refl ...
- 学习hadoop,搭建hadoop遇到一些特殊问题
我执行下面步骤: 1. 动态增加DataNode节点和TaskTracker节点 以host9为例 在host9上执行: 指定主机名 vi /etc/hostname 指定主机名到IP地址的映射 ...
- 【转】如何在 Eclipse 中進行 TFS 的版本管控
转自:http://www.dotblogs.com.tw/franma/archive/2010/05/04/15009.aspx 和上一篇一樣!所使用的版本也是 3.4 的 之前有被問到 Team ...
- C#学习笔记-----基于AppDomain的"插件式"开发
很多时候,我们都想使用(开发)USB式(热插拔)的应用,例如,开发一个WinForm应用,并且这个WinForm应用能允许开发人员定制扩展插件,又例如,我们可能维护着一个WinService管理系统, ...
- 字典树(Trie树)的实现及应用
>>字典树的概念 Trie树,又称字典树,单词查找树或者前缀树,是一种用于快速检索的多叉树结构,如英文字母的字典树是一个26叉树,数字的字典树是一个10叉树.与二叉查找树不同,Trie树的 ...
- 必须正确理解的---ng指令中的compile与link函数解析
这个绝对是深入的知识,但看完之后,对NG的理解就很利害啦. http://www.ifeenan.com/angularjs/2014-09-04-%5B%E8%AF%91%5DNG%E6%8C%87 ...
- poj 3252 组合数
主要考察组合数知识,初始化的时候参考公式 首先先推个公式,就是长度为len的Round Numbers的个数. 长度为len,第一位肯定是1了. 那么后面剩下 len-1位 ...
- Visual Studio一秒变Node.js IDE
(此文章同时发表在本人微信公众号"dotNET每日精华文章") 上个月微软发布了一个插件,可以让Visual Studio一秒变身最强大的Node.js开发工具.现在源代码移到了G ...
- LightOJ1057 Collecting Gold(状压DP)
这道题可以想到几点: 整个行程可以看作一次次的行走,每次行走都是用最短的路程从某一非空点到达另外一非空点: 两点间最少的步数是二者x和y坐标差的最大值: 返回原点这个过程,肯定是取完最后一个黄金后直接 ...
- 分布式实时日志处理平台ELK
这三样东西分别作用是:日志收集.索引与搜索.可视化展现 l logstash 这张架构图可以看出logstash只是collect和index的地方,运行时传入一个.conf文件,配置分三部分:in ...