iOS多线程(四):锁

多线程带来的问题之一就是安全问题,“锁”是为了使多个线程间可以相互排斥地使用全局变量等共享资源,简单来说就是保证同一时刻只有一个线程访问一块代码。

下面是一个有安全问题的代码例子:

- (void)test {
    //期望操作
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [self addNum];
    });
    //未预料的操作
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [self addNum];
    });
}

static int i = 0;

- (void)addNum {
    NSLog(@"执行开始 i=%d, 线程:%@", i, [NSThread currentThread]);
    I++;
    [NSThread sleepForTimeInterval:1];  //让线程阻塞一秒
    NSLog(@"执行结束 i=%d, 线程:%@", i, [NSThread currentThread]);
}

你只需要在你的主线程runloop中调用test就能测试了(如果目前看不明白关于线程的操作没关系,这里只是实现两个任务并行,并且模拟延长了addNum这个方法的执行时间),我们看到打印结果:

执行开始 i=0, 线程:<NSThread: 0x600000466140>{number = 5, name = (null)}
执行开始 i=0, 线程:<NSThread: 0x600000466940>{number = 4, name = (null)}
执行结束 i=2, 线程:<NSThread: 0x600000466940>{number = 4, name = (null)}
执行结束 i=2, 线程:<NSThread: 0x600000466140>{number = 5, name = (null)}

看到了么,我们无法预料异常的调用何时开始,我们期望得到的i = 1(我们的期望线程是number=4),然而却不一定能正常得到。所以,我们需要通过技术来实现共享变量的安全读写。

在介绍iOS的几种锁之前,先科普“死锁”的概念。

死锁:多线程等待一个永远无法实现的条件而无法继续执行。

1、NSLock

NSLock的使用非常简单,只需要将需要加锁的代码全部放进lockunlock方法中。
我们修改上面的addNum代码如下:

//注意lock是一个NSLock类的全局变量lock=[NSLock new]

- (void)addNum {
    [lock lock];
    NSLog(@"执行开始 i=%d, 线程:%@", i, [NSThread currentThread]);
    I++;
    [NSThread sleepForTimeInterval:1];  //让线程阻塞一秒
    NSLog(@"执行结束 i=%d, 线程:%@", i, [NSThread currentThread]);
    [lock unlock];
}

同样运行程序调用test方法,打印如下:

执行开始 i=0, 线程:<NSThread: 0x600000672940>{number = 5, name = (null)}
执行结束 i=1, 线程:<NSThread: 0x600000672940>{number = 5, name = (null)}
执行开始 i=1, 线程:<NSThread: 0x60400047e580>{number = 6, name = (null)}
执行结束 i=2, 线程:<NSThread: 0x60400047e580>{number = 6, name = (null)}

作用一目了然吧,这里我们的期望线程为number=5它完整的走了执行开始和执行结束,而没有受到number=6线程的干扰(因为如果获取不到锁,number=6线程就休眠了)。

注意一NSLock是基于POSIX线程实现的,lockunlock都必须在同一个线程执行。
注意二:我们尽量将lockunlock写在一起,如果业务需要导致获得锁和解锁的逻辑很分散(或者无法判断是否在同一线程),可以调用-(BOOL)tryLock方法尝试能否获取该锁,方便我们做不同的逻辑,代码如下:

- (void)addNum {
    if ([lock tryLock]) {
        NSLog(@"得到锁");
        NSLog(@"执行开始 i=%d, 线程:%@", i, [NSThread currentThread]);
        I++;
        [NSThread sleepForTimeInterval:1];  //让线程阻塞一秒
        NSLog(@"执行结束 i=%d, 线程:%@", i, [NSThread currentThread]);
        [lock unlock];
    } else {
        NSLog(@"没有得到锁");
    }
}

2、NSConditionLock

顾名思义,带条件的锁。同样实现了NSLocking协议,所以它的玩儿法和NSLock很像,而且它有一个方法很有意思。

  • lockWhenCondition:当线程A进入这里的时候,调用该方法,若不满足条件,线程就会进入休眠;若满足条件,就会得到锁并且执行下面的code。并且,若当前NSConditionLockcondition变为了满足的时候,线程A又会苏醒继续执行。当然,需要和unlockWithCondition结合使用。

下面就是一个实现多个并发任务同步执行的例子:

    NSConditionLock *clock = [[NSConditionLock alloc] initWithCondition:0];

    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [clock lockWhenCondition:0];
        NSLog(@"任务1");
        [NSThread sleepForTimeInterval:1];
        [clock unlockWithCondition:2];
    });
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [clock lockWhenCondition:1];
        NSLog(@"任务2");
        [NSThread sleepForTimeInterval:1];
        [clock unlockWithCondition:3];
    });
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [clock lockWhenCondition:2];
        NSLog(@"任务3");
        [NSThread sleepForTimeInterval:1];
        [clock unlockWithCondition:1];
    });

打印如下:

任务1
任务3
任务2

解析:我们三个任务中都使用了lockWhenCondition方法(注意这里我用的是“任务”而非“线程”,“使用GCD线程的分配不是我们要关心的”),我们初始化锁的时候用condition=0,所以先走任务1,任务1执行完毕调用[clock unlockWithCondition:2];所以接着走任务3,同理最后走任务2。

注意:如果大量使用条件锁导致线程休眠,而开辟了过多的线程,将会对性能造成消耗,所以使用需谨慎。

3、NSRecursiveLock

我们修改一下之前的例子,当线程1获取到“锁”过后,再次调用lock

- (void)addNum {
    [lock lock];
    [lock lock];
    NSLog(@"得到锁");
    NSLog(@"执行开始 i=%d, 线程:%@", i, [NSThread currentThread]);
    I++;
    [NSThread sleepForTimeInterval:1];  //让线程阻塞一秒
    NSLog(@"执行结束 i=%d, 线程:%@", i, [NSThread currentThread]);
    [lock unlock];
}

运行结果就是造成了“死锁”。还有一种常见的“死锁”的情况是:A线程获取到a锁,B线程获取到了b锁,同一时刻,A线程想要获取b锁,B线程想要获取a锁,A、B线程就会同时进入休眠(这就尴尬了)。

为了解决以上代码重复获取锁造成死锁的情况,我们引入了递归锁NSRecursiveLock。修改代码如下:

//注意:recursiveLock是一个NSRecursiveLock类型的全局变量,recursiveLock = [NSRecursiveLock new]

- (void)addNum {
    [recursiveLock lock];
    [recursiveLock lock];
    NSLog(@"得到锁");
    NSLog(@"执行开始 i=%d, 线程:%@", i, [NSThread currentThread]);
    I++;
    [NSThread sleepForTimeInterval:1];  //让线程阻塞一秒
    NSLog(@"执行结束 i=%d, 线程:%@", i, [NSThread currentThread]);
    [recursiveLock unlock];
    [recursiveLock unlock];
}

完美运行

注意一:使用NSRecursiveLock的时候,同样得注意获取锁和解锁需要一一对应,即一个lock对应一个unlock,不然锁不会处于可获取状态(哈哈,严谨吧,这里是不可获取状态而不是释放状态)。
注意二:若你能确定排除被重复加锁的情况,使用NSLock性能会更好。

4、@synchronized

@synchronized使用方法很简单,修改代码如下:

  • (void)addNum {
    @synchronized(self){
    NSLog(@"得到锁");
    NSLog(@"执行开始 i=%d, 线程:%@", i, [NSThread currentThread]);
    I++;
    [NSThread sleepForTimeInterval:1]; //让线程阻塞一秒
    NSLog(@"执行结束 i=%d, 线程:%@", i, [NSThread currentThread]);
    }
    }

@synchronized(obj),obj参数一般指定为互斥锁要保护的对象。值得提出的是,如果同一个obj对象做为了不同@synchronized(){}的参数,则这些代码块不能同时执行。

使用@synchronized我们不用管何时获取锁、何时释放锁,更容易的实现互斥,代码更加直观清晰;但是在性能上略显不足,而且实现并行算法有些复杂,不过这些都不能影响它极高的使用率。

下面盗一张ibireme的图,来看看各种锁的性能(图中的dispatch和pthread的锁有兴趣可以了解下。图的来源:https://blog.ibireme.com

image

写在后面

“锁”从来都不是为了炫技而生,大量使用锁不仅会带来性能问题,还会让代码更加的晦涩难懂,请大家合理的使用锁?。

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

推荐阅读更多精彩内容

  • 1.解决信号量丢失和假唤醒 public class MyWaitNotify3{ MonitorObject m...
    Q罗阅读 862评论 0 1
  • 引用自多线程编程指南应用程序里面多个线程的存在引发了多个执行线程安全访问资源的潜在问题。两个线程同时修改同一资源有...
    Mitchell阅读 1,962评论 1 7
  • demo下载 建议一边看文章,一边看代码。 声明:关于性能的分析是基于我的测试代码来的,我也看到和网上很多测试结果...
    炸街程序猿阅读 767评论 0 2
  • 进程和线程 进程 所有运行中的任务通常对应一个进程,当一个程序进入内存运行时,即变成一个进程.进程是处于运行过程中...
    胜浩_ae28阅读 5,080评论 0 23
  • 线程安全是怎么产生的 常见比如线程内操作了一个线程外的非线程安全变量,这个时候一定要考虑线程安全和同步。 - (v...
    幽城88阅读 637评论 0 0