Objective-c 中的锁

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Objective-c 中的锁相关的知识,希望对你有一定的参考价值。

参考技术A 1.Mutex 互斥锁.互斥锁同一时间只运行同一个线程操作,如果当一个线程正在持有锁,其他的线程想要持有锁,其他的线程会被阻塞,直到当前持有锁的线程释放该锁

   pthread_mutexattr_t mutexattr;

    pthread_mutexattr_init(&mutexattr);

    int re = pthread_mutex_init(&_mutex, &mutexattr);

  __block int value = 0;

    for (int i = 0; i<999; i++)

        dispatch_async(dispatch_get_global_queue(0, 0), ^

            pthread_mutex_lock(&_mutex);

            value++;

            pthread_mutex_unlock(&_mutex);

        );

   

    pthread_mutex_destroy(&_mutex);

    pthread_mutexattr_destroy(&mutexattr);

 ps:NSLock其实就是封装的pthread_mutex_lock

2.Recursive lock 递归锁.递归锁是互斥锁的一个变种.递归锁运行一个线程多次加锁,其他线程请求锁的时候同样会阻塞直到持有锁的线程释放锁.递归锁加锁次数和解锁次数必须一一对应,否则会无法释放锁.一般递归锁用在递归函数内部加锁.

3.Read-write lock 读写锁.读写锁一般用来大量的读写操作的时候.当你想要写操作时,必须等到读操作完成,释放读的锁,当你想要读的操作时,必须等到写的操作完成释放写的锁.

pthread_rwlockattr_init(&_rwAttr);

  int re = pthread_rwlock_init(&_rwLock, &_rwAttr);

    dispatch_async(dispatch_get_global_queue(0, 0), ^

        pthread_rwlock_wrlock(&_rwLock);

        NSLog(@"rwLock fro Writing%@",[NSThread currentThread]);

        sleep(3);

        NSLog(@"unlock fro Writing");

        pthread_rwlock_unlock(&_rwLock);

    );

    dispatch_async(dispatch_get_global_queue(0, 0), ^

        pthread_rwlock_rdlock(&_rwLock);

        NSLog(@"rwLock fro Reading%@",[NSThread currentThread]);

        sleep(2);

        NSLog(@"unlock fro Reading");

        pthread_rwlock_unlock(&_rwLock);

    );

    pthread_rwlockattr_destroy(&_rwAttr);

    pthread_rwlock_destroy(&_rwLock);

4.Distributed lock 分散锁.分散锁提供了一个在进程级别的互斥操作.分散锁不会阻塞其他的进程.它仅仅想想要上锁的进程报告当前锁的状态是busy.让进程觉得如何继续处理

ps:NSDistributedLock一般用于多个APP调度操作文件系统的文件

5.Spin lock 自旋锁.自旋锁通过不停的轮询条件来控制线程的操作.自旋锁的轮询机制使得其比他的的阻塞操作效率要高.但是注意,自旋锁已经不安全了,不要使用.

6.NSConditionLock 条件锁.条件锁是通过一个条件来使用mutex加锁,解锁的.一般情况下,条件锁使用的情况是当线程执行任务需要在指定条件下执行时.比较经典的就是生产者和消费者

NSConditionLock * condition = [[NSConditionLock alloc]initWithCondition:lockValue];

    dispatch_async(dispatch_get_global_queue(0, 0), ^

        [condition lockWhenCondition:lockValue];

        NSLog(@"productor lock%@",[NSThread currentThread]);

        sleep(2);

        NSLog(@"productor unlock");

        [condition unlockWithCondition:unlockValue];

    );

    dispatch_async(dispatch_get_global_queue(0, 0), ^

        [condition lockWhenCondition:unlockValue];

        NSLog(@"consumer lock%@",[NSThread currentThread]);

        sleep(3);

        NSLog(@"consumer unlock");

        [condition lockWhenCondition:lockValue];

    );

7.NSCondition . 条件相当于一个锁和一个临界条件的结合.锁会阻塞其他线程,保护数据,条件会保证在特定条件下执行任务.如果条件为假则会阻塞线程直到signal 发出之后,确定条件为真.

7.Deadlocks和Livelocks

deadLocks :任何时候当一个线程想要在同一时间获取多个锁都有可能发生死锁.当两个不同的线程都持有对方所需要的锁(加锁的资源),并且尝试获取另一个线程持有的锁(加锁的资源),就会发生死锁,因为每个线程都会被阻塞,无法获取对方的锁

liveLocks:类似于死锁,两个不同的线程竞争同一个资源时,有一个线程先放弃自己的锁,想要获取第二个锁,一旦它获取到第二个锁之后她又回头获取第一个自己放弃的锁.此时线程就被锁定了,因为线程一直重复的获取锁.

避免deadLocks 和 liveLocks 的方法就是保证同一时间只有一个锁.假如不能避免有多个锁,那就应该保证其他的线程不要去抢占这个锁.

ps:Core Foundation 是线程安全的.但是Foundation不是全部线程安全的,想要知道哪些类是线程安全的请参考 Thread Safety Summary

以上是关于Objective-c 中的锁的主要内容,如果未能解决你的问题,请参考以下文章

Objective-C中的Hello World

Objective-C中的@property和@synthesize用法

Objective-C中的方法重载?

Objective-C中的SEL (转载)

Objective-C中的引用计数

Swift 中的 Objective-C 协议