iOS-线程安全

本文转自掘金iOS-线程安全

多线程技术对于计算机开发带来了巨大的性能提升,同样也来带了新的伤痛——线程安全问题。在开发中,稍不注意,我们就可能写出让代码陷入不安全的境地,线程锁是一种用来帮助我们保护临界资源的手段。事实上,现代语言提供了多种不同的线程锁来保护代码。通过深入挖掘,可以发现线程锁的核心无非是Compare and Set,基于这简单的核心,衍化出了多种安全方案,本文就来讲讲锁的原理。

数据破坏

在理解如何保障临界代码的安全之前,我们需要了解数据为什么在多线程环境下被破坏。以简单的i++为例,这句代码将i自增一次,在编译成汇编代码后实际上会有三步操作:

movl -0x24(%rbp), %r8d
addl $0x1, %r8d
movl %r8d, -0x24(%rbp)

完成一次i++总共分为三步(把大象放进冰箱):

  • 取出i存放到临时寄存器上

  • 对寄存器的值+1

  • 将计算后的值存放回i的内存

image.png

假设线程A执行i++这句代码,在完成将计算后的数值存储回i的内存之前,线程B也开始执行这句代码,最终的结果是两次i++之后,值仍然为1,这时候数据就发生了破坏。如果这时候使用线程锁,那么B就会等待A完成存储数据后才能执行:

image.png

加锁后,B在执行i++之前会检测指令是否被锁住。如果被锁住,则开始休眠,直到A完成操作后被唤醒继续执行代码。这时候i++在多线程环境下是安全的

原子性

所谓原子操作是指不会被线程调度机制打断的操作;这种操作一旦开始,就一直运行到结束,中间不会有任何 context switch

我用原子性来表示某段多线程环境下的安全代码。从上面的代码来说,加锁之后的i++是具有原子性,因为代码在执行的过程中是线程安全的。此外,单条汇编一般也可以认为是具有原子性。而具有原子性的汇编指令也可以称作原子操作

原子(atom)指化学反应不可再分的基本微粒,原子在化学反应中不可分割

正常来说,代码分割到最小的单位就是单句汇编指令,比如上面的addl $0x1 %r8d可以被当做是代码中的原子。之所以说单条汇编是原子操作是因为在多线程环境下,汇编是不可再分割的,所以不会出现上面的破坏执行次序的问题。但是这并不是绝对的,比如系统中断可以中止正在执行的命令,这时候%r8d仍然是可能被意外修改。在汇编语言层面上,提供了LOCK指令前缀来保护指令执行过程层中的数据安全:

lock addl $0x1 %r8d

除此之外,在80486指令集中还有xaddcmpxchgxchg等指令是多处理器安全的。加了lock修饰的单条编译指令以及这些特殊的安全指令才算是真正的原子操作

线程锁

单条的汇编指令可以通过lock来保证原子操作,有通过锁住地址总线的方式保证指令执行过程中的读取安全的手段。当然,鉴于笔者水平,也不多做深究。

而在非汇编指令的代码层面上来说,我们使用互斥锁自旋锁条件锁等等工具来保护代码安全。那么这些锁具体是怎么实现的呢?锁分为信号量互斥锁,他们两者的使用区别如下:

  • 互斥锁

    互斥锁应当是排它的,意思是锁在被某个线程获取之后,只有获取锁的线程才能释放这个锁。其他线程必须等到获取锁的线程不再拥有锁之后,才能继续执行。在我使用NSLock的测试中,发现可以unlock其他线程的锁,因此严格来说NSLock并不适合被称作互斥锁

  • 信号量

    信号量拥有比互斥锁更多的用途。当信号量的value大于0时,所有的线程都能访问临界资源。在线程进入临界区后,value减一,反之亦然。如果信号量初始化为0时,可以看做是等待任务执行完成而非资源保护。value的操作应当是采用原子操作来保证指令的安全性的

互斥

锁的实现方式之一是互斥方式实现的(想了半天,还是决定用这个词)。即当线程B访问已经加锁了的临界资源时,检测到代码加锁,于是切换至内核态进行进一步的操作。伪代码大致实现如下,假定下面的代码是线程安全的:

if (!lock.try_lock()) {
    /// 切换至内核态
    thread current = this;
    list queue = get_global_wait_list();
    queue.push(current);
    current.sleep(forever);
}

此时,线程会进行休眠状态避免继续占用CPU资源,然后等待锁持有者执行完成释放锁。一旦任务完成,会检测是否存在等待执行代码的线程,如果存在,唤醒继续执行任务:

list queue = get_global_wait_list();
if ((t = queue.pop())) {
    t.wakeup();
}

在具体实现中互斥的实现要复杂的多,但是不妨碍它基于一个简单的机制实现。互斥的实现涉及到了可能发生的内核态切换,线程休眠、唤醒等,如果临界执行代码足够小而快,互斥的线程锁可能并不是最佳的实践方案

自旋

自旋的实现要比互斥简单的多。对于自旋实现的线程锁来说,存在一个线程间共享的标记变量。当某个线程进入临界区后,变量被标记,此时其他线程再想进入临界区,会进入while循环中空转等待:

while(flag) {
    continue;
}

自旋的实现逻辑足够简单,只要标记位的修改被设计为原子操作,就能保证多线程环境下的安全。对比互斥方案,自旋没有线程切换、休眠唤醒的开销。但是空转的代码会导致CPU在等待期间是满负荷执行的,如果加锁的代码不够小而快,甚至会直接影响到程序的运行

信号

信号的性能在自旋和互斥之间,通常的性能表现总是仅次于自旋。这里基于GCD的信号量实现来看,在进入等待时,会根据传入的超时时间出现三种表现:

  • DISPATCH_TIME_NOW

      while ((orig = dsema->dsema_value) < 0) {
          if (dispatch_atomic_cmpxchg2o(dsema, dsema_value, orig, orig + 1)) {
      #if USE_MACH_SEM
              return KERN_OPERATION_TIMED_OUT;
    
      #elif USE_POSIX_SEM || USE_FUTEX_SEM
              errno = ETIMEDOUT;
              return -1;
      #endif
          }
      }
    
    
  • DISPATCH_TIME_FOREVER

      #if USE_MACH_SEM
      do {
          kr = semaphore_wait(dsema->dsema_port);
      } while (kr == KERN_ABORTED);
      DISPATCH_SEMAPHORE_VERIFY_KR(kr);
    
      #elif USE_POSIX_SEM
      do {
          ret = sem_wait(&dsema->dsema_sem);
      } while (ret == -1 && errno == EINTR);
      DISPATCH_SEMAPHORE_VERIFY_RET(ret);
    
      #elif USE_FUTEX_SEM
      do {
          ret = _dispatch_futex_wait(&dsema->dsema_futex, NULL);
      } while (ret == -1 && errno == EINTR);
      DISPATCH_SEMAPHORE_VERIFY_RET(ret);
      #endif
    
    

根据超时时间的设置,信号量最终会表现为互斥或者自旋的方式实现,这也是为什么评测中信号量性能总是优于互斥低于自旋。虽然信号量的性能不是最优,但是这种结合方案保证了它的作用范围更大

barrier

barrier的任务总是保证在执行过程中,并发队列中有且只有barrier的任务在执行。最初笔者一度认为barrier的操作不过是加锁实现,后来在libdispatch的源码中得以窥见真容:

void dispatch_barrier_async_f(dispatch_queue_t dq, void *ctxt,
    dispatch_function_t func)
{
    dispatch_continuation_t dc;

    dc = fastpath(_dispatch_continuation_alloc_cacheonly());
    if (!dc) {
        return _dispatch_barrier_async_f_slow(dq, ctxt, func);
        }

    dc->do_vtable = (void *)(DISPATCH_OBJ_ASYNC_BIT | DISPATCH_OBJ_BARRIER_BIT);
    dc->dc_func = func;
    dc->dc_ctxt = ctxt;

    _dispatch_queue_push(dq, dc);
}

相比dispatch_async的实现,barrier只是简单的将任务标记为DISPATCH_OBJ_ASYNC_BIT。但在执行队列任务的_dispatch_queue_drain会循环获取任务并且判断,barrier任务的真正实现在这个函数中。由于函数实现稍长,笔者只放上去除额外参数的伪代码:

void _dispatch_queue_drain() {
    while((task = queue.next())) {
        if (queue.excute_barrier()) {
            return;
        } else if (task.do_vtable & DISPATCH_OBJ_ASYNC_BIT) {
          return;
        } else {
           task.execute();
        }
    }
}

当循环取出队列任务执行的时候,检测到当前存在barrier的任务,则停止任务获取,直到当前所有的任务执行完成。并且在barrier执行过程中,不允许执行其他任务

Compare and Set

上面总结了很多种线程锁方案,包括从伪代码和源代码窥探实现,线程锁的实现机制其实基于很简单的概念:标志是否被占用。而在这其中,核心确实无非Compare and Set,这两个是最核心的操作,通过原子操作实现这两个步骤来保证多线程锁的获取中不会出现另外的线程安全问题。笔者用八个字总结了线程锁的特性:

因为简单,所以可靠

最后

从入职新东家以来,深感到经济基础决定上层建筑这句话的意义。基础薄弱影响了笔者难以突破很多技术上的关口,是以未来很长一段时间都要将自己曾经欠下的债慢慢补上。同时,规范化的开发流程也是对自己的一个巨大挑战。我想,仅仅是实现功能就能自称工程师的话,无疑显得廉价。如何构建更稳健的代码,学会编写应对异常环境下的代码,等到那时候,我才有资格自称工程师

参考

互斥锁的实现

用汇编实现原子操作

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

推荐阅读更多精彩内容