iOS中的多种锁(Lock)

在查看SDWebImage这种第三方的时候会发现有些地方有用到锁,其中最常见的就是@ synchronized,所以今天我们就由点到面的来了解一下iOS的各种锁。下面我们用脑图来分析下锁的各种类型:
各种锁.png

问题场景:

开发数中会难免会遇到多线程竞争资源的问题,从而带来了线程安全的问题。
所谓线程安全:当一个线程访问数据的时候,其他线程不能对其访问,直至该线程访问完毕。简单来讲就是同一时刻对同一个数据操作的线程只有一个。只有确保了这样,才能使数据不会被其他线程影响。
那么,我们怎样保证线程安全呢?
此时,锁就派上用场了,可以确保同一时刻只有同一个线程对同一个数据源进行访问。

1、@synchronized锁

@synchronized(美 [,sɪnkrənaɪ'zeʃən]:同步)是OC层面的锁,synchronized block 与 [_lock lock] & [_lock unlock] 效果相同,但语法更加简洁可读,但代价是性能的降低。
官网介绍:防止不同的线程同时获取相同的锁。
知识网址:http://yulingtianxia.com/blog/2015/11/01/More-than-you-want-to-know-about-synchronized/
重点:@synchronized 结构在工作时为传入的对象分配了一个递归锁。所谓递归锁是在被同一个线程重复调用时不会产生死锁。NSRecursiveLock(递归锁)类也是这样的,我们后面会有分析。
特殊情况:
1、你调用 sychronized 的每个对象,Objective-C runtime 都会为其分配一个递归锁并存储在哈希表中。
2、如果在 sychronized 内部对象被释放或被设为 nil 看起来都 OK。不过这没在文档中说明,所以我不会再生产代码中依赖这条。
3、注意不要向你的 sychronized block 传入 nil!这将会从代码中移走线程安全。你可以通过在 objc_sync_nil 上加断点来查看是否发生了这样的事情。

代码如下:
- (void)synchronizedLock{
    /*
     NSMutableArray *_elements;
     _elements在任何情况下都只会在一个线程中运行
     */
    @synchronized(_elements){
        [_elements addObject:@"1"];
    };
}

2、NSLock锁

原理:NSLock实现了最基本的互斥锁,遵循NSLocking协议,通过lock和unLock来进行锁定于解锁。当一个线程访问的时候,该线程获得锁,其他线程访问的时候,将被操作系统挂起,直到该线程释放锁,其他线程才能对其进行访问,从而确保线程安全。如果连续锁定,则会造成死锁问题。

代码如下:
/*
 A :lock的最简单使用
 */
- (void)initLock{
    //1、对锁进行初始化
    _elements = [NSMutableArray array];
    _lock = [[NSLock alloc] init];
}

- (void)push:(id)element{
    //2、上锁
    [_lock lock];
    [_elements addObject:element];
    //3、解锁
    [_lock unlock];
    
}
/*
 B :lock的结合GCD多线程调用使用
 */
- (void)GDCAndLock{
    _lock = [[NSLock alloc]init];
    
    //在多个线程中调用。由于使用锁的线程锁是没有执行完毕的,所以其他显线程不能调用,直到执行完毕后,才允许其他线程调用。
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        NSLog(@"1");
        [self lockFounction:[NSThread currentThread] num: 1];
    });
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        NSLog(@"2");
        [self lockFounction:[NSThread currentThread] num: 2];
    });
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        NSLog(@"3");
        [self lockFounction:[NSThread currentThread] num: 3];
    });
    
  
}
- (void)lockFounction:(NSThread *)thread num:(NSInteger) num {
    [_lock lock];
    NSLog(@"thread - %@, num - %ld", thread, num);
    sleep(5);
    [_lock unlock];
}

打印如图:


打印.png
/*
 C :lock的tryLock和lockBeforeDate两个方法的使用。
 tryLock方法会尝试加锁,如果锁不可用(已经被锁住),则并不会阻塞线程,并返回NO。lockBeforeDate:方法会在所指定Date之前尝试加锁,如果在指定时间之前都不能加锁,则返回NO。
 */

- (void)tryLockAndDate{
    _lock = [[NSLock alloc]init];
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        //lockBeforeDate会在指定的时间之前加锁,所以已经使用过[_lock lock]了.下面相当于在当前时间之前上锁了。
        [_lock lockBeforeDate:[NSDate date]];
        NSLog(@"1需要线程同步的操作1 开始");
        sleep(2);
        NSLog(@"1需要线程同步的操作1 结束");
        [_lock unlock];
    });
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        sleep(1);
        if ([_lock tryLock]) {//尝试获取锁,如果获取不到返回NO,不会阻塞该线程
            NSLog(@"2锁可用的操作");
            [_lock unlock];
        }else{
            NSLog(@"2锁不可用的操作");
        }
        NSDate *date = [[NSDate alloc] initWithTimeIntervalSinceNow:3];
        if ([_lock lockBeforeDate:date]) {
            //尝试在未来的3s内获取锁,并阻塞该线程,如果3s内获取不到恢复线程, 返回NO,不会阻塞该线程
            NSLog(@"2没有超时,获得锁");
            [_lock unlock];
        }else{
            NSLog(@"2超时,没有获得锁");
        }
    });
    
}

3、递归锁NSRecursiveLock

NSRecursiveLock递归锁可以被同一线程多次请求,但不会引起死锁。这主要是用在循环或者递归操作场景中。

- (void)useNSRecursiveLock{
    //如果使用_lock会招致死锁,因为被同一个线程多次调用。每次进入这个block时,都会去加一次锁,而从第二次开始,由于锁已经被使用了且没有解锁,所以它需要等待锁被解除,这样就导致了死锁,线程被阻塞住了。
//    _lock = [[NSLock alloc] init];
    NSRecursiveLock *lock = [[NSRecursiveLock alloc] init];
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        //创建一个静态方法,block方法
        static void (^RecursiveMethod)(int);
        RecursiveMethod = ^(int value) {
            [lock lock];
            if (value > 0) {
                NSLog(@"value = %d", value);
                sleep(1);
                RecursiveMethod(value - 1);
            }
            [lock unlock];
        };
        RecursiveMethod(5);//方法内部判断来执行5次
    });
}

4、NSConditionLock条件锁

当我们在使用多线程的时候,只有一把会lock和unlock的锁就不能满足我们的需要了。因为普通的锁只关心锁与不锁,但是并不在乎什么时候才能开锁,而在处理资源共享场景的时候,多数情况下只有满足一定条件下才能打开这把锁。(Condition:美 [kən'dɪʃən] 条件)
NSConditionLock实现步骤:
NSConditionLock实现了NSLocking协议,一个线程会等待另一个线程unlock或者unlockWithCondition:之后再走lock或者lockWhenCondition:之后的代码。
锁定和解锁的调用可以随意组合,也就是说 lock、lockWhenCondition:与unlock、unlockWithCondition: 是可以按照自己的需求随意组合的。
划重点:
1、只有 condition 参数与初始化时候的 condition 相等,lock 才能正确进行加锁操作。
2、unlockWithCondition: 并不是当 condition 符合条件时才解锁,而是解锁之后,修改 condition 的值。

/*
 在线程 1 解锁成功之后,线程 2 并没有加锁成功,而是继续等了 1 秒之后线程 3 加锁成功,这是因为线程 2 的加锁条件不满足,初始化时候的 condition 参数为 0,而线程 2
 加锁条件是 condition 为 1,所以线程 2 加锁失败。
 lockWhenCondition 与 lock 方法类似,加锁失败会阻塞线程,所以线程 2 会被阻塞着。
 tryLockWhenCondition: 方法就算条件不满足,也会返回 NO,不会阻塞当前线程。
 lockWhenCondition:beforeDate:方法会在约定的时间内一直等待 condition 变为 2,并阻塞当前线程,直到超时后返回 NO。
 */

代码:

- (void)nsconditionlock {
    NSConditionLock * cjlock = [[NSConditionLock alloc] initWithCondition:0];
    
    //1、线程 1 解锁成功
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [cjlock lock];
        NSLog(@"线程1加锁成功");
        sleep(1);//线程休眠一秒
        [cjlock unlock];
        NSLog(@"线程1解锁成功");
    });
    
    //2、初始化时候的 condition 参数为0,所以此处加锁失败,返回NO,此处线程阻塞。全部现成执行完毕后执行此处锁
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        sleep(1);//线程休眠一秒
        [cjlock lockWhenCondition:1];
        NSLog(@"线程2加锁成功");
        [cjlock unlock];
        NSLog(@"线程2解锁成功");
    });
    
    //3、tryLockWhenCondition尝试加锁  初始化时候的 condition 参数为0,所以此处加锁成功。方法就算条件不满足,也会返回 NO,不会阻塞当前线程。
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        sleep(2);
        
        if ([cjlock tryLockWhenCondition:0]) {
            NSLog(@"线程3加锁成功");
            sleep(2);
            /*
             A:成功案例
             这里会先解锁当前的锁,之后修改condition的值为100.在下一个condition为100的线程中会加解锁成功,如果下个锁中的condition等待的值不是100,那么就会导致加锁失败。
             */
            [cjlock unlockWithCondition:100];
            NSLog(@"线程3解锁成功");
            
            /*
             B:失败案例
             [cjlock unlockWithCondition:4];
             NSLog(@"线程3仍然会解锁成功,之后修改condition的值为4");
             */
            
        } else {
            NSLog(@"线程3尝试加锁失败");
        }
    });
    
    //4、lockWhenCondition:beforeDate:方法会在约定的时间内一直等待 condition 变为 2,并阻塞当前线程,直到超时后返回 NO。
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        if ([cjlock lockWhenCondition:100 beforeDate:[NSDate dateWithTimeIntervalSinceNow:10]]) {
            NSLog(@"线程100加锁成功");
            [cjlock unlockWithCondition:1];
            NSLog(@"线程100解锁成功");
        } else {
            NSLog(@"线程100尝试加锁失败");
        }
    });
}

5、NSCondition

定义及使用:NSCondition 是一种特殊类型的锁,通过它可以实现不同线程的调度。A线程被某一个条件所阻塞,直到B线程满足该条件,从而发送信号给A线程使得A线程继续执行,例如:你可以开启一个线程下载图片,一个线程处理图片。这样的话,需要处理图片的线程由于没有图片会阻塞,当下载线程下载完成之后,则满足了需要处理图片的线程的需求,这样可以给定一个信号,让处理图片的线程恢复运行。
重点:
1、NSCondition 的对象实际上作为一个锁和一个线程检查器,锁上之后,其他线程也能继续上锁,之后根据条件决定是否继续运行线程,如果线程进入waiting状态,当其他线程中的该锁执行signal(信号)或者broadcast(广播)时,线程被唤醒,继续运行该线程之后的方法。。
2、NSCondition 可以手动控制现成的挂起和唤醒,可以利用这个特性设置依赖。
特别提醒:
signal只是唤醒单个线程,broadcast唤醒所有的线程。
broadcast :广播 { 英 ['brɔːdkɑːst] 美 ['brɔdkæst]}

- (void)nsCondition {
    NSCondition * cjcondition = [NSCondition new];
    /*
     在加上锁之后,调用条件对象的 wait 或 waitUntilDate: 方法来阻塞线程,直到条件对象发出唤醒信号或者超时之后,再进行之后的操作。
     */
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        [cjcondition lock];
        NSLog(@"线程1线程加锁----NSTreat:%@",[NSThread currentThread]);
        [cjcondition wait];
        NSLog(@"线程1线程唤醒");
        [cjcondition unlock];
        NSLog(@"线程1线程解锁");
    });
    
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        [cjcondition lock];
        NSLog(@"线程2线程加锁----NSTreat:%@",[NSThread currentThread]);
        if ([cjcondition waitUntilDate:[NSDate dateWithTimeIntervalSinceNow:10]]) {
            NSLog(@"线程2线程唤醒");
            [cjcondition unlock];
            NSLog(@"线程2线程解锁");
        }
    });

    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        sleep(2);
        /*
1、休眠时间如果超过了线程中条件锁等待的时间,那么所有的线程都不会被唤醒。不管是哪一个线程中设置的时间,都不能超时,否则就会返回NO,全部不执行!切记切记!
2、一次只能唤醒一个线程,要调用多次才可以唤醒多个线程,如下调用两次,将休眠的两个线程解锁。
3、唤醒的顺序为线程添加的顺序。
*/
        
        [cjcondition signal];
        [cjcondition signal];

        //一次性全部唤醒
        //[cjcondition broadcast];
    });
}

6、dispatch_semaphore信号量

GCD的信号量机制实现锁,等待信号和发送信号。
1、dispatch_semaphore 是 GCD 用来同步的一种方式,与他相关的只有三个函数,一个是创建信号量,一个是等待信号,一个是发送信号。
2、dispatch_semaphore的机制就是当有多个线程进行访问的时候,只要有一个获得了信号,其他线程就必须等待该信号的释放。

 1、dispatch_semaphore 和 NSCondition 类似,都是一种基于信号的同步方式,但 NSCondition 信号只能发送,不能保存(如果没有线程在等待,则发送的信号会失效)。而 dispatch_semaphore 能保存发送的信号。dispatch_semaphore 的核心是 dispatch_semaphore_t 类型的信号量。
 2、dispatch_semaphore_create(1) 方法可以创建一个 dispatch_semaphore_t ( 英  ['seməfɔː])类型的信号量,设定信号量的初始值为 1。注意,这里的传入的参数必须大于或等于 0,否则 dispatch_semaphore_create 会返回 NULL。
 3、dispatch_semaphore_wait(semaphore, overTime); 方法会判断 semaphore 的信号值是否大于 0。大于 0 不会阻塞线程,消耗掉一个信号,执行后续任务。如果信号值为 0,该线程会和 NSCondition 一样直接进入 waiting状态,等待其他线程发送信号唤醒该线程执行后续任务,或者当 overTime 时限到了,也会执行后续任务。
 4、dispatch_semaphore_signal(semaphore); 发送信号,如果没有等待的线程接受信号,则使 signal 信号值加一(做到对信号的保存)。
 5、一个 dispatch_semaphore_wait(semaphore, overTime); 方法会去对应一个 dispatch_semaphore_signal(semaphore); 看起来像 NSLock 的 lock 和 unlock,其实可以这样理解,区别只在于有信号量这个参数,,lock unlock 只能同一时间,只能有一个线程访问被保护的临界区,而如果信号量参数初始值为 x,那么就会有 x 个线程可以同时访问被保护的临界区。
 */
- (void)useDispatch_semaphore {
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(1);
    // overTime设置为6秒
    dispatch_time_t overTime = dispatch_time(DISPATCH_TIME_NOW, 3 * NSEC_PER_SEC);
   
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        dispatch_semaphore_wait(semaphore, overTime);
        NSLog(@"线程1开始");
        sleep(5);
        NSLog(@"线程1结束");
        dispatch_semaphore_signal(semaphore);
    });
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        sleep(1);
        dispatch_semaphore_wait(semaphore, overTime);
        NSLog(@"线程2开始");
        dispatch_semaphore_signal(semaphore);
    });
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        sleep(1);
        dispatch_semaphore_wait(semaphore, overTime);
        NSLog(@"线程3开始");
        dispatch_semaphore_signal(semaphore);
    });
}

7、OSSpinLock自旋锁

首先导入:#import <libkern/OSAtomic.h>
OSSpinLock 是一种自旋锁,和互斥锁类似,都是为了保证线程安全的锁。但是这是不一样的:
互斥锁:当一个线程获得此锁之后,其他线程再获取将会被阻塞,直到该锁被释放。
自旋锁:当一个线程获得此锁之后,其他线程将会一直循环查看该锁是否被释放。锁比较适用于锁的持有者保存时间较短的情况下。
OSSpinLock自旋锁只有加锁、尝试加锁和解锁三个方法。

/*
 YY大神 @ibireme 的文章也有说这个自旋锁存在优先级反转问题,具体文章可以戳 不再安全的 OSSpinLock,而 OSSpinLock 在iOS 10.0中被 <os/lock.h> 中的 os_unfair_lock 取代。
常用的相关API:
 // 初始化
 os_unfair_lock_t unfairLock = &(OS_UNFAIR_LOCK_INIT);
 // 加锁
 os_unfair_lock_lock(unfairLock);
 // 尝试加锁
 BOOL b = os_unfair_lock_trylock(unfairLock);
 // 解锁
 os_unfair_lock_unlock(unfairLock);
 os_unfair_lock 用法和 OSSpinLock 基本一致,就不一一列出了。
 */
- (void)osspinlock {
    __block OSSpinLock theLock = OS_SPINLOCK_INIT;//在iOS10之后被ns_unfair_lock替换
    
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        OSSpinLockLock(&theLock);
        NSLog(@"线程1开始");
        sleep(3);
        NSLog(@"线程1结束");
        OSSpinLockUnlock(&theLock);
        
    });
    
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        OSSpinLockLock(&theLock);
        sleep(1);
        NSLog(@"线程2");
        OSSpinLockUnlock(&theLock);
        
    });
}

8、锁的总结

其实每一种锁基本上都是加锁、等待、解锁的步骤,理解了这三个步骤就可以帮你快速的学会各种锁的用法。
1、@synchronized 的效率最低,不过它的确用起来最方便,所以如果没什么性能瓶颈的话,可以选择使用 @synchronized。
2、当性能要求较高时候,可以使用 pthread_mutex 或者 dispath_semaphore,由于 OSSpinLock 不能很好的保证线程安全,而在只有在 iOS10 中才有 os_unfair_lock ,所以,前两个是比较好的选择。既可以保证速度,又可以保证线程安全。
3、对于 NSLock 及其子类,速度来说 NSLock < NSCondition < NSRecursiveLock < NSConditionLock 。

好了,要赶火车回家过年了,来年开春了在接着写啦。大家新春快乐!☺

参考资料:
http://www.jb51.net/article/127573.htm
http://blog.csdn.net/liupinghui/article/details/67637830
最后奉上本人总结的代码:https://github.com/caiqingchong/iOSLock

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

推荐阅读更多精彩内容

  • 一个人走得快,一个人到底能走多远? 《知识变现》这本书开启了的自由工作之路,主要讲述了自由工作者知识变现,建立个人...
    不忘初心_42f4阅读 468评论 0 0
  • “九”是一个奇数,古代以奇数为阳,偶数为阴。而“九”是最大的奇数,故视“九”为极阳数。古代帝王的宫殿多以九来装饰,...
    工程宝阅读 656评论 0 0
  • 在一个微信群里,我结识一位90后妹妹,她有自己的本职工作,在公号写作上几乎日更的速度,每天一篇原创文章,约3000...
    成长馨路阅读 484评论 0 1