iOS多线程

首先每个正在运行的应用程序都是一个进程,每个进程系统都会分配独立的内存资源。而一个进程中的所有任务都是在线程中执行的,所以每个进程至少有一个线程,这个线程也称为主线程,而如果想要并发执行多个任务那么就要开启多条线程,也称为多线程。多线程在一定意义上实现了进程内的资源共享,以及效率的提升。同时,在一定程度上相对独立,是执行任务最基本的单元,有自己栈和寄存器。而对应单核 CPU 来说,多线程并不是真正意义上并发执行任务,只是CPU快速地在多条线程之间调度,CPU调度线程的时间足够短,就造成了多线程并发执行的假象。就单核CPU而言多线程可以解决线程阻塞的问题,但是其本身效率并没有提高,多核CPU的并行才真正解决了运行效率问题。

上面这段抄的,来源在这


iOS 多线程方案

1. pthread 这个是一套 C 语言跨平台多线程 API,其产生的线程生命周期是需要手动管理

2. NSThread 是苹果对 pthread 面向对象的封装,和上面一样是需要手动管理线程生命周期

3. GCD 是苹果另一套多线程解决方案,其线程生命周期是自动管理,使用简单,也是最常使用的

4. NSOperation 是苹果基于 GCD 面向对象封装,提供了一实用的功能,例如设置线程最大并发数,添加线程间依赖等


GCD

首先在 GCD 中并不直接操作线程,而是通过其提供的任务管理方式队列和任务执行方式同步和异步,间接管理线程

队列作用:任务的执行方式

队列分为两种,一种是是串行队列,一种是并发队列

串行队列:一个任务完成后再执行下一个任务,根据先进先出的按顺序执行

并发队列:多个任务并发(同时)执行

特殊的队列:全局队列和主队列

全局队列是并发队列,而主队列是在主线程上,是串行队列

同步和异步作用:能不能开启新的线程

同步:在当前线程中执行任务,不具备开启新线程的能力

异步:在新的线程中执行任务,具备开启新线程的能力

只要是同步或主队列两者占一项,就不会开启新线程,串行执行任务;

只有异步并且不是主队列,才会开启新线程,根据其搭配的队列并发或串行是否执行任务

在 GCD 中使用 sync 同步函数往当前串行队列添加任务,就会产生死锁阻塞线程


常用函数 

dispatch_sync 同步执行任务

dispatch_async 异步执行任务

dispatch_once 只执行一次使当前添加的任务,并且是线程安全的,常用于创建单例对象

dispatch_apply 使当前添加的任务执行指定次数,这个相当于内置循环执行多次任务,这个函数会调用主线程执行任务,如果是任务耗时不要使用,使用手动设置循环调用多次异步并发队列执行耗时任务

dispatch_after 使任务延迟执行

dispatch_group_async 与 dispatch_group_notify 添加任务依赖,执行完前者的任务之后都会执行后者的任务

dispatch_group_wait 会阻塞当前线程,等待 group 中的任务执行完,之后再回到当前线程执行

dispatch_barrier_async 并发执行任务时,当前添加任务不会执行,执行当前任务时是串行执行,使用场景 IO的多读单写


线程同步方案

OSSpinLock

这个是自旋锁,等待锁的线程会处于忙等状态,一直占用 CPU 资源,从 iOS10开始,会出现警告,因为这个锁不再安全,可能会出现优先级反转的问题,如果等待的线程优先级较高,它会一直占用 CPU 资源,优先级低的线程就无法释放锁资源

os_unfair_lock

是 iOS10之后才支持的,是 apple 用来取代 OSSpinLock 一种线程同步方案,是互斥锁,和自旋锁区别在于等待锁资源的线程会处于休眠状态,并非忙等状态

一般开启多线程时,如果涉及到线程同步的说,不建议修改线程优先级,容易出现优先级反转的问题,特别是使用 OSSpinLock 时

一般来说,只要不是对耗时任务进行使用线程同步,那么使用自旋锁的效率和开销会优于互斥锁,因为线程执行任务的时间短,自旋锁会一直忙等,及时拿到锁资源,如果是互斥锁,因为线程执行任务的时间短,这种频繁休眠唤醒操作会增加资源开销,降低效率,如果是耗时任务线程同步,那么使用互斥锁优于自旋锁,因为自旋锁等待的线程处于激活状态一直访问锁资源,而另外拿到锁资源的线程此时还没给 CPU 调度到,所以此时耗时任务没有执行完,无法释放锁资源,此时会降低效率

两者 api 几乎一样,锁的初始化都是宏定义提供的,前者宏定义是0,后者是将 os_unfair_lock 结构体初始化

#define OS_SPINLOCK_INIT    0

#define OS_UNFAIR_LOCK_INIT ((os_unfair_lock){0})

提供的3个函数 lock unlock tryLock

tryLock 表示,当前是否能拿到锁资源,能就返回 YES,可以配合 if 一起使用

pthread_mutex

mutex 表示互斥锁,是一套 c 语言的跨平台线程同步 api

初始化是,可以选择锁的类型,一种是默认就是互斥锁,一种是递归锁

递归锁的特性是,不同线程间就跟互斥锁特性一样,但是可以让同一个线程多次获取锁资源,主要使用场景就是对一个递归方法进行线程同步

mutex 这套 api 还有个特点就是给线程添加条件 condition

他提供了三个函数

pthread_cond_wait() 接收两个参数,一个是条件,一个线程,可以让当前线程等待,因为是互斥锁 ,等待时会休眠,同时释放锁的资源,等待被当前条件重新激活,激活时会唤醒,线程重新拿到锁的资源

pthread_cond_signal() 会激活一个等待该条件的线程

pthread_cond_broadcast() 激活所有等待该条件的线程

NSLock

是 apple 对 mutex 互斥锁的面向对象封装 

NSRecursiveLock

是 apple 对 mutex 递归锁的面向对象封装 

NSCondition

是 apple 对 mutex 条件的面向对象封装 

NSConditionLock

这个 NSCondition 进一步的封装,这点在于可以设置具体的条件值,只有满足条件值的线程才能拿到锁的资源,在释放锁的资源时可以更改条件值

@synchronized

也是对 mutex 递归锁封装,@synchronized(obj) 其内部会生成 obj 对应的递归锁,然后进行加锁、解锁操作,但是性能不好,不推荐使用


GCD 提供线程同步方案

dispatch_semaphore

semaphore 也就是信号量,这个信息量可以用来控制线程并发访问量大数量

当设置信号量为1,代表同时只允许1条线程访问资源,保证线程同步

提供 3个函数

dispatch_semaphore_create(value) 初始化信息号

dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER) 如果信息量的值 <=0,当前线程就会进入休眠等待,直到信息量的值 >0,如果信息量的值 >0 就减 1,让线程往下执行后面的代码

dispatch_semaphore_signal(semaphore) 让信息号加1

wait 和 signal 和之前 condition 条件很像

DISPATCH_QUEUE_SERIAL

直接使用 GCD 的串行队列,也是可以实现线程同步的


读写安全方案-多读单写

pthread_rwlock 读写锁

dispatch_barrier_async 异步栅栏函数 

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

推荐阅读更多精彩内容