Swift多线程开发 - 4. 多线程开发的问题

多线程并发有三个比较突出的问题,

  • 资源竞争
  • 死锁
  • 优先级倒置

1. 资源竞争

在不同线程的同时想要更新一个变量,而读写是分开的,那么就可能会出现资源竞争的情况。
CPU是基于时钟信号来工作的, 每个周期只能执行一个单一的操作

iPhone XS 处理器 2.49 GHz, 意味着一秒钟可以执行 2,490,000,000 个周期

例子

线程1和2同时想要更新count

count += 1

这行代码仔细拆分开来会是这样子

  1. 加载count的值到内存中
  2. 在内存中将count+1
  3. 将新的count值写入磁盘中
资源竞争

图片引自 Raywenderlich.com 的 Concurrency by tutorial

如上图所示,

  • 在第一个时钟周期,线程1读取到了count的值
  • 在第二个时钟周期,线程1在内存中更新了count的值,线程2读取到了count的值
  • 在第三个时钟周期,线程1把更新后的值2写入了磁盘,而线程2此时在内存中更新值1为2
  • 在第四个时钟周期,线程2把更新后的值2写入了磁盘

此时,count的值是2而不是3,

多核处理器可以同时处理多个线程,但都是有一个时钟频率驱动工作

事实上,例子中线程1只要在两个时钟周期之前更新就不会有问题了,而现在的iPhone每秒可以执行几十亿次(当然并不是所有写入操作都能这么快完成的)。
所以在实际的项目中,特别是频繁多线程写入读取的时候,会有可能发生这种资源竞争问题。在测试的时候怎么跑都是对的,而放到生产环境中就又出问题了。

解决方案

一般资源竞争问题可以通过

  1. 串行线程访问
  2. Thread Barrier
  3. 线程锁

a. 串行线程访问

对于简单读写操作来说,串行Dispatch可以说是最简单高效的方案了

private let threadSafeCountQueue = DispatchQueue(label: "...") // 默认串行线程
private var _count = 0
public var count: Int {
    get {
        return threadSafeCountQueue.sync {
            _count
        }
    } set {
        threadSafeCountQueue.sync {
            _count = newValue
        }
    } 
}

lazy初始化同样不是线程安全的,如果会有多线程访问并初始化的情况,也可以给这个lazy变量加一个串行线程来使其线程安全

b. Thread Barrier

Thread Barrier

图片引自 Raywenderlich.com 的 Concurrency by tutorial

实现
private let threadSafeCountQueue = DispatchQueue(label: "...",  attributes: .concurrent)
private var _count = 0 
public var count: Int {
    get {
        return threadSafeCountQueue.sync {
            return _count
    }
} set {
    threadSafeCountQueue.async(flags: .barrier) { [unowned self] in
     self._count = newValue }
    } 
}

Barrier里的代码会在之前任务执行完之后才会继续,
一旦Barrier触发了,它所在的线程就会像变成串行队列一样,后续的任务要在Barrier的内容执行完之后,才会继续照旧并发运行。

c. 线程锁

基本上所有任务都可以通过上次两种方案解决,直接使用线程锁容易产生问题。
详细可以参考我的另一片文章[Swift] iOS线程锁

2. 死锁

比如你正在开一辆车,双向个一条车道,你临时要调个头。
也就是要等对向车道的车过了之后有空才能转个弯,所以你这条车道后面的车都被你堵住了。
而你没有办法转过去,因为此时对面车道的车也要往你的车道上转弯,但它必须等你走了,让你后面的车走了,它才能转进去。
这时的情况就是死锁。

也就是两个线程或者两个任务结束/开始的条件相互依赖而导致无法进行到下一步

典型的就是在主线程中DispatchQueue.main.sync(){}

Swfit本身的机制保障了使得我们很少会遇到这种问题,除非我们自己手动调用Semaphore或者其它线程锁。所以在使用对应的机制的时候经过严谨的测试或者至少深思熟虑一些即可避免这个问题

3. 优先级反转

优先级反转发生在优先级(QoS)较低的队列比优先级较高的队列有更高的系统优先级时。
在本系列的第二篇文章中有提到,将高优先级的任务提交到较低优先级的队列,系统会提升队列的优先级来匹配任务的优先级。
这会导致比如仅有一个.userInteractive级任务的.utility级队列甚被系统排到了.userInitiated前面执行。

解决方案: 仅将相同优先级的任务加入一个队列,其它任务加入到其它队列。

实际上,发生优先级反转更常见的情况是,较高优先级的队列与较低优先级的队列共享资源。当较低的队列锁定该对象时,较高的队列现在必须等待。在释放锁之前,高优先级队列实际上一直处于阻塞状态,而低优先级任务在运行时不执行任何操作。



作者博客地址

系列文章链接

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

推荐阅读更多精彩内容