iOS开发-10.多线程

  • 1.iOS中的常见多线程方案
    image
a) NSThread / GCD / NSOperation底层都是pthread

b) NSThread开启线程方式
    1) 动态实例化
NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(test:) object:nil];
thread.threadPriority = 1;  //设置线程的优先级(0.0 - 1.0, 1.0最高级)
[thread start];
    2) 静态实例化
[NSThread detachNewThreadSelector:@selector(test:) toTarget:self withObject:nil];
    3) 隐式实例化
        3.1)
[self performSelectorOnMainThread:@selector(test:) withObject:nil waitUntilDone:YES];
  • 2.GCD的常用函数
a) GCD中有2个用来执行任务的函数
    1) 用同步的方式执行任务
    dispatch_sync(dispatch_queue_t queue, dispatch_block_t block); // queue:队列 block:任务
        1.1) dispatch_sync : 立马在当前线程同步执行任务,不执行就不会往下走
    
    2) 用异步的方式执行任务
    dispatch_async(dispatch_queue_t queue, dispatch_block_t block);
        2.1) dispatch_async : 不要求立马在当前线程同步执行任务,等上一个任务(也可能是整个外部的大函数执行完毕)执行完了再执行

b) GCD源码https://github.com/apple/swift-corelibs-libdispatch
  • 3.GCD的队列
a) GCD的队列可以分为2大类型
    1) 并发队列(Concurrent Dispatch Queue)
        1.1) 可以让多个任务并发(同时)执行(自动开启多个线程同时执行任务)
        1.2) 并发功能只有在异步(dispatch_async)函数下才有效

b) 串行队列(Serial Dispatch Queue)
    1) 让任务一个接着一个地执行(一个任务执行完毕后,再执行下一个任务)
    2) 主队列是一种特殊的串行队列
    
c) 队列的特点 : 排队,FIFO,First In First Out 先进先出

d) Global Queue全局队列的指针地址是相同的,而手动创建的队列则地址是不相同的
  • 4.容易混淆的术语
a) 同步和异步主要影响:能不能开启新的线程
    1) 同步:在当前线程中执行任务,不具备开启新线程的能力
    2) 异步:在新的线程中执行任务,具备开启新线程的能力

b) 并发和串行主要影响:任务的执行方式
    1) 并发:多个任务并发(同时)执行
    2) 串行:一个任务执行完毕后,再执行下一个任务
  • 5.各种队列的执行效果
    image
a) 使用sync函数往当前串行队列中添加任务,会卡住当前的串行队列(产生死锁)

b) performSelector:withObject:afterDelay: 
    1) 有afterDelay的方法都是跟Runloop有关,相当于添加了一个定时器到Runloop去执行 所以你要看当前执行任务的线程有没有Runloop
    2) 在主线程好时正常调用,但是会延迟,因为主线程默认就会开启一个Runloop,延迟是因为主线程的任务执行顺序是串行的,需要等上一个任务执行完毕,可以理解为包装该线程调用的{}范围函数调用完毕,例如viewDidLoad方法调用完毕
    3) 但是在子线程却不好使,直接不调用,因为子线程没有开启Runloop来保活
  • 6.队列组的使用
a) 思考:如何用gcd实现以下功能
    1) 异步并发执行任务1、任务2
    2) 等任务1、任务2都执行完毕后,再回到主线程执行任务3
image
  • 7.多线程的安全隐患
a) 资源共享
    1) 1块资源可能会被多个线程共享,也就是多个线程可能会访问同一块资源
    2) 比如多个线程访问同一个对象、同一个变量、同一个文件

b) 当多个线程访问同一块资源时,很容易引发数据错乱和数据安全问题

c) 多线程安全隐患示例01 – 存钱取钱
image
d) 多线程安全隐患示例02 – 卖票
image
e) 多线程安全隐患分析
image
f) 多线程安全隐患的解决方案
    1) 解决方案:使用线程同步技术(同步,就是协同步调,按预定的先后次序进行)
    2) 线程同步本质:是不能同时让多条线程占用一个资源
    2) 常见的线程同步技术是:加锁(所有线程都用同一把锁)
    3) 锁的原理:首先判断这把锁有没有被人加过,没有就会加锁,有被加过,就会等待
image
  • 8.iOS中的线程同步方案
a) OSSpinLock(High-level-lock高级锁)
    1) OSSpinLock叫做”自旋锁”,等待锁的线程会处于忙等(busy-wait)状态(相当于执行了一个do while循环)一直占用着CPU资源
    2) iOS10.0开始不推荐使用,目前已经不再安全,可能会出现优先级反转问题(如果等待锁的线程优先级较高,它会一直占用着CPU资源,优先级低的线程就无法释放锁,造成死锁的现象)
    3) 需要导入头文件#import <libkern/OSAtomic.h>
image
b) os_unfair_lock(Low-level-lock低级锁)
    1) os_unfair_lock用于取代不安全的OSSpinLock ,从iOS10开始才支持
    2) 从底层调用看,等待os_unfair_lock锁的线程会处于休眠状态,并非忙等
    3) 需要导入头文件#import <os/lock.h>
image
c) pthread_mutex(Low-level-lock低级锁)
    1) mutex叫做”互斥锁”,等待锁的线程会处于休眠状态
    2) 需要导入头文件#import <pthread.h>
image
    3) pthread_mutex – 递归锁
        3.1) 递归锁:允许用同一条线程对一把锁进行重复加锁
image
    4) pthread_mutex – 条件
        4.1) 条件(等待条件,等不到条件就休眠,等待的时候解锁,唤醒的时候加锁,次数对等,一般用于线程之间的通信)
image
d) NSLock
    1) NSLock是对mutex普通锁的封装
image
e) NSRecursiveLock
    1) NSRecursiveLock是对mutex递归锁的封装,API跟NSLock基本一致
f) NSCondition
    1) NSCondition是对mutex和cond的封装
    2) signal:信号发出的时机,决定了后续代码的执行,因为在解锁前和解锁后
    3) signal:信号(单个)和broadcast:广播(多个)的区别
image
g) NSConditionLock
    1) NSConditionLock是对NSCondition的进一步封装,可以设置具体的条件值
    2) 初始化不设置条件值的时候,默认条件值就是0
    3) lock:直接加锁,不等条件值,如果没有加锁,那就加锁
    4) lockWhenCondition:条件值为多少的时候才加锁
    5) unlockWithCondition:解锁并且设置条件值
    6) 在子线程执行的任务,可以达到依赖的效果
image
h) dispatch_semaphore
    1) semaphore叫做”信号量”
    2) 信号量的初始值,可以用来控制线程并发访问的最大数量
    3) 信号量的初始值为1,代表同时只允许1条线程访问资源,保证线程同步
image
i) dispatch_queue
    1) 直接使用GCD的串行队列,也是可以实现线程同步的
    2) 不是说有多线程就需要加锁
image
j) @synchronized
    1) @synchronized是对mutex递归锁的封装
    2) 源码查看:objc4中的objc-sync.mm文件
    3) @synchronized(obj)内部会生成obj对应的递归锁,然后进行加锁、解锁操作
    4) 性能比较差,苹果官方不推荐使用,所以代码提示也没有
image
  • 9.iOS线程同步方案性能比较
    image
  • 10.自旋锁、互斥锁比较
a) 什么情况使用自旋锁比较划算?
    1) 预计线程等待锁的时间很短
    2) 加锁的代码(临界区)经常被调用,但竞争情况很少发生
    3) CPU资源不紧张
    4) 多核处理器

b) 什么情况使用互斥锁比较划算?
    1) 预计线程等待锁的时间较长
    2) 临界区有IO操作
    3) 临界区代码复杂或者循环量大
    4) 临界区竞争非常激烈
    5) 单核处理器
    
c) 临界区:lock与unlock之间的代码 

d) IO操作:文件操作(读和写的操作)
  • 11.atomic
a) atomic用于保证属性setter、getter的原子性操作,相当于在getter和setter内部加了线程同步的锁

b) 可以参考源码objc4的objc-accessors.mm

c) 它并不能保证使用属性的过程是线程安全的
  • 12.iOS中的读写安全方案
a) 思考如何实现以下场景
    1) 同一时间,只能有1个线程进行写的操作
    2) 同一时间,允许有多个线程进行读的操作
    3) 同一时间,不允许既有写的操作,又有读的操作

b) 上面的场景就是典型的“多读单写”,经常用于文件等数据的读写操作

c) iOS中的实现方案有
    1) pthread_rwlock:读写锁
image
    2) dispatch_barrier_async:异步栅栏调用
        2.1) 这个函数传入的并发队列必须是自己通过dispatch_queue_cretate创建的
        2.2) 如果传入的是一个串行或是一个全局的并发队列,那这个函数便等同于dispatch_async函数的效果
image
  • 13.GNUstep
a) GNUstep是GNU计划的项目之一,它将Cocoa的OC库重新开源实现了一遍

b) 源码地址:http://www.gnustep.org/resources/downloads.php

c) 虽然GNUstep不是苹果官方源码,但还是具有一定的参考价值
  • 14.其它知识点总结
a) lldb查看汇编指令级别代码时一行一行的运行 
    1) stepi(si)
    2) nexti(ni):也是一行一行,但是遇到函数调用会一笔带过这个函数调用
    3) c:直接跳到下一个断点
    
b) 定义变量的时候同时给他赋值结构体可以这么干,但是不能直接给结构体赋值

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

推荐阅读更多精彩内容

  • 身为开发者的我们,总是会面临着各种各样的需求.必不可少的一个硬性规则:从用户体验的角度来开发项目!那么这就要求我...
    Small_Potato阅读 464评论 2 8
  • 进程 1.进程是指在系统中正在运行的一个应用程序2.每个进程之间是相互独立的,每个进程均运行在其专用且受保护的内存...
    hmj1993阅读 1,723评论 1 3
  • GCD是Grand Central Dispatch的缩写,有人叫它大中枢派发、或者大中央调度,不管叫啥,总之,它...
    xfs_coder阅读 471评论 0 1
  • 一、进程与线程 1.进程: 系统中正在运行的一个应用程序,每个进程之间是独立的,每个进程均运行在其专用且受保护的内...
    云上的彩阅读 683评论 0 0
  • 一、多线程NSThread 1.多线程基础知识 线程与进程的关系(1). 线程是CPU执行任务的基本单位,一个进程...
    IOS_龙阅读 592评论 0 8