6.iOS开发之多线程

iOS日常开发几乎都离不开多线程,基本上常用的几个NSThread、GCD和NSOperation了,另外其实还有基于C语言的pthread,POSIX接口。日常整理一下下。

线程的概念

每个应用程序都是一个独立的进程,一个进程的所有任务都在线程中进行,因此一个进程中至少有一个线程。线程是程序执行流的最小单元,是进程中的一个实体,是执行程序最基本的单元,有自己栈和寄存器。

什么是多线程

多线程就是一个进程开启多条线程,让所有任务并发执行。一定意义上实现了进程内的资源共享,以及效率的提升。

Pthreads && NSThread

POSIX线程(POSIX threads),简称Pthreads,是线程的POSIX标准。该标准定义了创建和操纵线程的一整套API。在类Unix操作系统(Unix、Linux、Mac OS X等)中,都使用Pthreads作为操作系统的线程。

简单来说就是基于C的跨平台的操纵线程API。需要自己管理线程的生命周期。

NSThread

苹果提供的基于OC的API,封装了Pthreads,需要自己管理线程的生命周期,通常用的最多的是[NSThread currentThread]获取当前线程。

GCD(大中央调度)

封装了线程操作,提供简洁的C语言接口,使用简单高效,苹果官方推荐使用。自动管理了线程的生命周期以及任务的执行规则。

任务与队列

任务:就是执行操作的意思,在线程中执行的那段代码,GCD 是放在 block 中的。

常用的任务执行方法:

dispatch_sync同步与dispatch_async异步任务执行

两者的主要区别是:是否等待队列的任务执行结束,以及是否具备开启新线程的能力。

队列:统一管理任务的方式,队列都是按照FIFO的原则依次触发任务。分为串行队列与并行队列。队列只负责任务的调度,而不负责任务的执行,任务是在线程中执行的。(可以理解成任务是放在队列里面的,要被调度到线程中去执行)

串行队列(DISPATCH_QUEUE_SERIAL):所有任务会在一条线程中执行(同步任务则是当前线程,异步任务则新开辟线程),并且一个任务执行完毕后,才开始执行下一个任务。(等待完成)

主队列是系统创建的串行队列,任务都在主线程上执行。

并行队列(DISPATCH_QUEUE_CONCURRENT):可以开启多条线程并行执行任务(异步任务新开辟线程,同步任务不开启新线程),并且当一个任务放到指定线程开始执行时,下一个任务就可以开始执行了。(等待发生)

全局队列是系统创建好的一个并行队列,使用起来与自己创建的并行队列无本质差别。

串行队列里面有一个任务中调用同步方法就会造成死锁

经典例子:主队列中viewDidLoad任务调用sync就会死锁crash

关于多线程的同步异步任务与串行并行队列组合:同步在任何队列都不会开启新线程,在主线程会造成死锁,异步在并行可开启多条线程,在串行只开启一条线程,在主队列不开启线程。

GCD其他函数

1. dispatch_after 用于任务延时执行

2. dispatch_once 保证函数在整个生命周期内只会执行一次(多用于单例)

3. dispatch_group 队列组(调度组) dispatch_group_async添加任务 全部执行完调用dispatch_group_notify通知

dispatch_group_wait会阻塞当前线程

dispatch_group_enter标志着一个任务追加到 group,执行一次,相当于 group 中未执行完毕任务数 +1

dispatch_group_leave标志着一个任务离开了 group,执行一次,相当于 group 中未执行完毕任务数 -1。

当 group 中未执行完毕任务数为0的时候,才会使 dispatch_group_wait 解除阻塞,以及执行追加到 dispatch_group_notify 中的任务。

使用dispatch_group_enter与dispatch_group_leave的方式为多,例如dispatch_group_async里执行的是异步代码,这种方式会等待异步任务完成而用dispatch_group_async则不会。

4.dispatch_barrier_sync(堵塞当前队列与主队列中的所有任务) dispatch_barrier_async(只堵塞当前队列的任务)栅栏函数

必须是自定义的并发队列 必须是同一个队列  【保证顺序执行 保证线程安全】

5. dispatch_apply 重复执行某个任务,任务队列是并行队列则重复执行的任务会并发执行;为串行队列则任务会顺序执行;该函数为同步函数,要防止线程阻塞和死锁

6.dispatch_semaphore信号量

dispatch_semaphore_create创建信号量,初始值不能小于0;

dispatch_semaphore_wait等待降低信号量,也就是信号量-1;

dispatch_semaphore_signal提高信号量,也就是信号量+1;

dispatch_semaphore_wait和dispatch_semaphore_signal通常配对使用。

NSOperation && NSOperationQueue

苹果对于GCD的封装,NSOperation其实就是上面所说的任务,NSOperationQueue就是类似于GCD中的队列,用于管理任务。

NSOperation

抽象类,不能直接使用,我们要用他的两个子类NSBlockOperation与NSInvocationOperation。或者自定义子类重写main方法来实现相关功能。

NSOperationQueue

Operation加入队列之后不用调用任务的start方法,队列会管理任务的执行情况。

支持优先级(queuePriority),支持线程之间的依赖关系(addDependency)。

支持 KVO,可以监测 operation 是否正在执行(isExecuted)、 是否结束(isFinished)、是否取消(isCanceld)。

maxConcurrentOperationCount 控制并发数【队列中同时可调度任务】为1时并非为串行队列,队列任务的执行顺序依然取决于很多因素。

suspended队列可以挂起

cancelAllOperations取消队列中所有任务,取消后得重新添加任务

线程锁

多线程操作是多个线程并行的,所以同一块资源可能在同一时间被多个线程访问,所以为了线程安全,需要线程锁。自旋锁(会忙等)与互斥锁(会休眠),

OSSpinLock (自旋锁)会忙等,效率高,cpu消耗大

效率高 目前已经不再线程安全 不推荐使用

pthread_mutex(互斥锁) 会休眠

基于Linux的一套跨平台线程锁API,非常底层,可以创建条件锁,递归锁等。 使用递归锁能防止死锁,因为递归锁能够对同一把锁重复加锁。

dispatch_semaphore (信号量)

性能好,推荐使用

NSLock (pthread_mutex的一层封装) 对象锁

lock与unlock

@synchronized  递归锁

底层封装的pthread_mutex的PTHREAD_MUTEX_RECURSIVE 模式,锁对象来表示是否为同一把锁。

要传入一个同步对象(一般就是self),然后将你需要加锁的资源放入代码块中,如果该资源有线程正在访问时,会让其他线程等待

推荐使用pthread_mutex 与 dispatch_semaphore 性能高

多线程的优化思路

1、减少队列切换:过多的上下文切换会带来资源开销。

2、控制线程数量:GCD 中并行队列并不能限制线程数量,可以创建多个串行队列来模拟并行的效果。

3、线程优先级权衡:qualityOfService

iOS 8 过后设置队列优先级的方法如下:

dispatch_queue_attr_t attr=dispatch_queue_attr_make_with_qos_class(DISPATCH_QUEUE_CONCURRENT,QOS_CLASS_BACKGROUND,0);

dispatch_queue_t queue=dispatch_queue_create("x.x.x",attr);

这里就设置了一个QOS_CLASS_BACKGROUND优先级,比较适合后台异步下载大文件之类的业务。

4、主线程任务的优化

(1)内存复用 比如UITableViewCell 的复用

(2)懒加载任务: UI 组件必须在主线程初始化,那么就需要用时再初始化。

(3)任务拆分排队执行

通过监听 Runloop 即将结束等通知,将大量的任务拆分开来,在每次 Runloop 循环周期执行少量任务。其实在实践这种优化思路之前,应该想想能不能将任务放到异步线程,而不是用这种比较极端的优化手段。

(4)  主线程空闲时执行任务



文章参考或引用:

iOS多线程编程

『GCD』详尽总结

iOS 如何高效的使用多线程

iOS 任务调度器:为 CPU 和内存减负

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容