iOS底层原理 多线程之安全锁

转链接:https://juejin.im/post/5d395318f265da1b8608ca98

自旋锁 OS_SPINLOCK

什么是优先级反转

简单从字面上来说,就是低优先级的任务先于高优先级的任务执行了,优先级搞反了。那在什么情况下会生这种情况呢?

假设三个任务准备执行,A,B,C,优先级依次是A>B>C;

首先:C处于运行状态,获得CPU正在执行,同时占有了某种资源;

其次:A进入就绪状态,因为优先级比C高,所以获得CPU,A转为运行状态;C进入就绪状态;

第三:执行过程中需要使用资源,而这个资源又被等待中的C占有的,于是A进入阻塞状态,C回到运行状态;

第四:此时B进入就绪状态,因为优先级比C高,B获得CPU,进入运行状态;C又回到就绪状态;

第五:如果这时又出现B2,B3等任务,他们的优先级比C高,但比A低,那么就会出现高优先级任务的A不能执行,反而低优先级的B,B2,B3等任务可以执行的奇怪现象,而这就是优先反转。

OS_SPINLOCK叫做自旋锁,等待锁的进程会处于忙等(busy-wait)状态,一直占用着CPU资源,目前已经不安全,可能会出现优先级翻转问题。

OS_SPINLOCKAPI

//初始化 一般是0,或者直接数字0也是ok的。
#define OS_SPINLOCK_INIT    0
//锁的初始化
OSSpinLock lock = OS_SPINLOCK_INIT;
//尝试加锁
bool ret = OSSpinLockTry(&lock);
//加锁
OSSpinLockLock(&lock);
//解锁
OSSpinLockUnlock(&lock);
复制代码

OSSpinLock简单实现12306如何卖票

//基类实现的卖票
- (void)__saleTicket{
    NSInteger oldCount = self.ticketsCount;
    if (isLog) {
        sleep(sleepTime);
    }
    oldCount --;
    self.ticketsCount = oldCount;
    if (isLog) {
    printf("还剩% 2ld 张票 - %s \n",(long)oldCount,[NSThread currentThread].description.UTF8String);
    }

}

- (void)ticketTest{
    self.ticketsCount = 10000;
    NSInteger count = self.ticketsCount/3;
    dispatch_queue_t queue = dispatch_queue_create("tick.com", DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(queue, ^{
        if (time1 == 0) {
            time1 = CFAbsoluteTimeGetCurrent();
        }
        for (int i = 0; i < count; i ++) {
            [self __saleTicket];
        }
    });

    dispatch_async(queue, ^{
        if (time1 == 0) {
            time1 = CFAbsoluteTimeGetCurrent();
        }
        for (int i = 0; i < count; i ++) {
            [self __saleTicket];
        }
    });
    dispatch_async(queue, ^{
        if (time1 == 0) {
            time1 = CFAbsoluteTimeGetCurrent();
        }
        for (int i = 0; i < count; i ++) {
            [self __saleTicket];
        }
    });
    dispatch_barrier_async(queue, ^{
        CFAbsoluteTime time = CFAbsoluteTimeGetCurrent() - time1;
        printf("tick cost time:%f",time);
    });
}
- (void)__getMonery{
    OSSpinLockLock(&_moneyLock);
    [super __getMonery];
    OSSpinLockUnlock(&_moneyLock);
}
- (void)__saleTicket{
    OSSpinLockLock(&_moneyLock);
    [super __saleTicket];
    OSSpinLockUnlock(&_moneyLock);
}
- (void)__saveMonery{
    OSSpinLockLock(&_moneyLock);
    [super __saveMonery];
    OSSpinLockUnlock(&_moneyLock);
}

- (void)__saleTicket{
    NSInteger oldCount = self.ticketsCount;
    oldCount --;
    self.ticketsCount = oldCount;
}
//log
还剩 9 张票 - <NSThread: 0x600003dc6080>{number = 3, name = (null)} 
还剩 8 张票 - <NSThread: 0x600003dc6080>{number = 3, name = (null)} 
还剩 7 张票 - <NSThread: 0x600003dc6080>{number = 3, name = (null)} 
还剩 6 张票 - <NSThread: 0x600003df3a00>{number = 4, name = (null)} 
还剩 5 张票 - <NSThread: 0x600003df3a00>{number = 4, name = (null)} 
还剩 4 张票 - <NSThread: 0x600003df3a00>{number = 4, name = (null)} 
还剩 3 张票 - <NSThread: 0x600003dc0000>{number = 5, name = (null)} 
还剩 2 张票 - <NSThread: 0x600003dc0000>{number = 5, name = (null)} 
还剩 1 张票 - <NSThread: 0x600003dc0000>{number = 5, name = (null)} 
复制代码

汇编分析

for (NSInteger i = 0; i < 5; i ++) {
    [[[NSThread alloc]initWithTarget:self selector:@selector(__saleTicket) object:nil] start];
}

然后将睡眠时间设置为600s,方便我们调试。
- (void)__saleTicket{
    OSSpinLockLock(&_moneyLock);//此行打断点
    [super __saleTicket];
    OSSpinLockUnlock(&_moneyLock);
}
复制代码

到了断点进入Debug->Debug WorkFlow ->Always Show Disassembly,到了汇编界面,在LLDB输入stepi,然后一直按enter,一直重复执行上句命令,直到进入了循环,就是类似下列的三行,发现ja跳转到地址0x103f3d0f9,每次执行到ja总是跳转到0x103f3d0f9,直到线程睡眠结束。

->  0x103f3d0f9 <+241>: movq   %rcx, (%r8)
0x103f3d0fc <+244>: addq   $0x8, %r8
0x103f3d100 <+248>: cmpq   %r8, %r9
0x103f3d103 <+251>: ja     0x103f3d0f9
复制代码

可以通过汇编分析了解到自旋锁是真的忙等,闲不住的锁。

os_unfair_lock

os_unfair_lock被系统定义为低级锁,一般低级锁都是闲的时候在睡眠,在等待的时候被内核唤醒,目的是替换已弃用的OSSpinLock,而且必须使用OS_UNFAIR_LOCK_INIT来初始化,加锁和解锁必须在相同的线程,否则会中断进程,使用该锁需要系统在__IOS_AVAILABLE(10.0),锁的数据结构是一个结构体

OS_UNFAIR_LOCK_AVAILABILITY
typedef struct os_unfair_lock_s {
    uint32_t _os_unfair_lock_opaque;
} os_unfair_lock, *os_unfair_lock_t;
复制代码

os_unfair_lock使用非常简单,只需要在任务前加锁,任务后解锁即可。

@interface FYOSUnfairLockDemo : FYBaseDemo
@property (nonatomic,assign) os_unfair_lock lock;
@end

@implementation FYOSUnfairLockDemo
- (instancetype)init{
    if (self = [super init]) {
        self.lock = OS_UNFAIR_LOCK_INIT;
    }
    return self;
}

- (void)__saveMonery{
    os_unfair_lock_lock(&_unlock);
    [super __saveMonery];
    os_unfair_lock_unlock(&_unlock);
}
- (void)__getMonery{
    os_unfair_lock_lock(&_unlock);
    [super __getMonery];
    os_unfair_lock_unlock(&_unlock);
}
- (void)__saleTicket{
    os_unfair_lock_lock(&_unlock);
    [super __saleTicket];
    os_unfair_lock_unlock(&_unlock);
}
@end
//log
还剩 9 张票 - <NSThread: 0x600002eb4bc0>{number = 3, name = (null)} 
还剩 8 张票 - <NSThread: 0x600002eb4bc0>{number = 3, name = (null)} 
还剩 7 张票 - <NSThread: 0x600002eb4bc0>{number = 3, name = (null)} 
还剩 6 张票 - <NSThread: 0x600002eb1500>{number = 4, name = (null)} 
还剩 5 张票 - <NSThread: 0x600002eb1500>{number = 4, name = (null)} 
还剩 4 张票 - <NSThread: 0x600002eb1500>{number = 4, name = (null)} 
还剩 3 张票 - <NSThread: 0x600002ed4340>{number = 5, name = (null)} 
还剩 2 张票 - <NSThread: 0x600002ed4340>{number = 5, name = (null)} 
还剩 1 张票 - <NSThread: 0x600002ed4340>{number = 5, name = (null)} 
复制代码

汇编分析

LLDB中命令stepi遇到函数会进入到函数,nexti会跳过函数。我们将断点打到添加锁的位置

- (void)__saleTicket{
    os_unfair_lock_lock(&_unlock);//断点位置
    [super __saleTicket];
    os_unfair_lock_unlock(&_unlock);
}
复制代码

执行si,一直enter,最终是停止该位子,模拟器缺跳出来了,再enter也没用了,因为线程在睡眠了。syscall是调用系统函数的命令。

libsystem_kernel.dylib`__ulock_wait:
    0x107a3b9d4 <+0>:  movl   $0x2000203, %eax          ; imm = 0x2000203 
    0x107a3b9d9 <+5>:  movq   %rcx, %r10
->  0x107a3b9dc <+8>:  syscall
复制代码

互斥锁 pthread_mutex_t

mutex叫互斥锁,等待锁的线程会处于休眠状态。

-(void)dealloc{
    pthread_mutex_destroy(&_plock);
    pthread_mutexattr_destroy(&t);
}
-(instancetype)init{
    if (self =[super init]) {
        //初始化锁的属性 
//      pthread_mutexattr_init(&t);
//      pthread_mutexattr_settype(&t, PTHREAD_MUTEX_NORMAL);
//      //初始化锁
//      pthread_mutex_init(&_plock, &t);

        pthread_mutex_t plock = PTHREAD_MUTEX_INITIALIZER;
        self.plock = plock;
    }
    return self;
}
-(void)__saleTicket{
    pthread_mutex_lock(&_plock);
    [super __saleTicket];
    pthread_mutex_unlock(&_plock);
}
- (void)__getMonery{
    pthread_mutex_lock(&_plock);
    [super __getMonery];
    pthread_mutex_unlock(&_plock);
}
- (void)__saveMonery{
    pthread_mutex_lock(&_plock);
    [super __saveMonery];
    pthread_mutex_unlock(&_plock);
}
//log

还剩 9 张票 - <NSThread: 0x6000014e3600>{number = 3, name = (null)} 
还剩 8 张票 - <NSThread: 0x6000014c8d80>{number = 4, name = (null)} 
还剩 7 张票 - <NSThread: 0x6000014c8f40>{number = 5, name = (null)} 
还剩 4 张票 - <NSThread: 0x6000014c8f40>{number = 5, name = (null)} 
还剩 3 张票 - <NSThread: 0x6000014c8f40>{number = 5, name = (null)} 
还剩 5 张票 - <NSThread: 0x6000014c8d80>{number = 4, name = (null)} 
还剩 6 张票 - <NSThread: 0x6000014e3600>{number = 3, name = (null)} 
还剩 2 张票 - <NSThread: 0x6000014c8d80>{number = 4, name = (null)} 
还剩 1 张票 - <NSThread: 0x6000014e3600>{number = 3, name = (null)} 
复制代码

互斥锁有三个类型

/*
 * Mutex type attributes
 */
 普通锁
#define PTHREAD_MUTEX_NORMAL        0
//检查错误
#define PTHREAD_MUTEX_ERRORCHECK    1
//递归锁
#define PTHREAD_MUTEX_RECURSIVE     2
//普通锁
#define PTHREAD_MUTEX_DEFAULT       PTHREAD_MUTEX_NORMAL
复制代码

当我们这样子函数调用函数会出现死锁的问题,这是怎么出现的呢?第一把锁是锁住状态,然后进入第二个函数,锁在锁住状态,在等待,但是这把锁需要向后执行才会解锁,到时无限期的等待。

- (void)otherTest{
    pthread_mutex_lock(&_plock);
    NSLog(@"%s",__func__);
    [self otherTest2];
    pthread_mutex_unlock(&_plock);
}
- (void)otherTest2{
    pthread_mutex_lock(&_plock);
    NSLog(@"%s",__func__);
    pthread_mutex_unlock(&_plock);
}

//log
-[FYPthread_mutex2 otherTest]
复制代码

上面这个需求需要使用两把锁,或者使用递归锁来解决问题。

- (void)otherTest{
    pthread_mutex_lock(&_plock);
    NSLog(@"%s",__func__);
    [self otherTest2];
    pthread_mutex_unlock(&_plock);
}
- (void)otherTest2{
    pthread_mutex_lock(&_plock2);
    NSLog(@"%s",__func__);
    pthread_mutex_unlock(&_plock2);
}

//log
-[FYPthread_mutex2 otherTest]
-[FYPthread_mutex2 otherTest2]
复制代码

从使用2把锁是可以解决这个问题的。 递归锁是什么锁呢?允许同一个线程对一把锁重复加锁。

NSLock、NSRecursiveLosk

NSLock是对mutex普通锁的封装

使用(LLDB) si可以跟踪[myLock lock];的内部函数最终是pthread_mutex_lock

Foundation`-[NSLock lock]:
    0x1090dfb5a <+0>:  pushq  %rbp
    0x1090dfb5b <+1>:  movq   %rsp, %rbp
    0x1090dfb5e <+4>:  callq  0x1092ca3fe               ; symbol stub for: object_getIndexedIvars
    0x1090dfb63 <+9>:  movq   %rax, %rdi
    0x1090dfb66 <+12>: popq   %rbp
->  0x1090dfb67 <+13>: jmp    0x1092ca596   ;
//  symbol stub for: pthread_mutex_lock
复制代码

NSLock API大全

//协议NSLocking
@protocol NSLocking

- (void)lock;
- (void)unlock;

@end

@interface NSLock : NSObject <NSLocking> {
@private
    void *_priv;
}
- (BOOL)tryLock;//尝试加锁
- (BOOL)lockBeforeDate:(NSDate *)limit;//在某个日期前加锁,
@property (nullable, copy) NSString *name API_AVAILABLE(macos(10.5), ios(2.0), watchos(2.0), tvos(9.0));
@end
复制代码

用法也很简单

@interface FYNSLock(){
    NSLock *_lock;
}
@end

@implementation FYNSLock
- (instancetype)init{
    if (self = [super init]) {
        //封装了mutex的普通锁
        _lock=[[NSLock alloc]init];
    }
    return self;
}

- (void)__saveMonery{
    [_lock lock];
    [super __saveMonery];
    [_lock unlock];
}
- (void)__saleTicket{
    [_lock lock];
    [super __saleTicket];
    [_lock unlock];
}
- (void)__getMonery{
    [_lock lock];
    [super __getMonery];
    [_lock unlock];
}
@end
//log

还剩 9 张票 - <NSThread: 0x600003d4dc40>{number = 3, name = (null)} 
还剩 8 张票 - <NSThread: 0x600003d4dc40>{number = 3, name = (null)} 
还剩 7 张票 - <NSThread: 0x600003d4dc40>{number = 3, name = (null)} 
还剩 6 张票 - <NSThread: 0x600003d7bfc0>{number = 4, name = (null)} 
还剩 5 张票 - <NSThread: 0x600003d7bfc0>{number = 4, name = (null)} 
还剩 4 张票 - <NSThread: 0x600003d7bfc0>{number = 4, name = (null)} 
还剩 3 张票 - <NSThread: 0x600003d66c00>{number = 5, name = (null)} 
还剩 2 张票 - <NSThread: 0x600003d66c00>{number = 5, name = (null)} 
还剩 1 张票 - <NSThread: 0x600003d66c00>{number = 5, name = (null)} 
复制代码

NSRecursiveLock也是对mutex递归锁的封装,APINSLock基本一致

- (BOOL)tryLock;//尝试加锁
- (BOOL)lockBeforeDate:(NSDate *)limit;//日期前加锁
复制代码

递归锁可以对相同的线程进行反复加锁

@implementation FYRecursiveLockDemo
- (instancetype)init{
    if (self = [super init]) {
        //封装了mutex的递归锁
        _lock=[[NSRecursiveLock alloc]init];
    }
    return self;
}
- (void)otherTest{
    static int count = 10;
    [_lock lock];
    while (count > 0) {
        count -= 1;
        printf("循环% 2d次 - %s \n",count,[NSThread currentThread].description.UTF8String);
        [self otherTest];
    }
    [_lock unlock];
}
@end

//log
循环 9次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 8次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 7次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 6次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 5次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 4次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 3次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 2次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 1次 - <NSThread: 0x60000274e900>{number = 1, name = main} 
循环 0次 - <NSThread: 0x60000274e900>{number = 1, name = main}
复制代码

NSCondition 条件

- (void)wait;//等待
- (BOOL)waitUntilDate:(NSDate *)limit;
- (void)signal;//唤醒一个线程
- (void)broadcast;//唤醒多个线程
复制代码

NSCondition是对mutexcond的封装

- (instancetype)init{
    if (self = [super init]) {
        //遵守的 lock协议 的 条件🔐
        _lock=[[NSCondition alloc]init];
        self.array =[NSMutableArray array];
    }
    return self;
}
- (void)otherTest{
    [[[NSThread alloc]initWithTarget:self selector:@selector(__remove) object:nil] start];
    [[[NSThread alloc]initWithTarget:self selector:@selector(__add) object:nil] start];
}
- (void)__add{
    [_lock lock];
    [self.array addObject:@"Test"];
    NSLog(@"添加成功");
    sleep(1);
    [_lock signal];//唤醒一个线程
    [_lock unlock];
}
- (void)__remove{
    [_lock lock];
    if (self.array.count == 0) {
        [_lock wait];
    }
    [self.array removeLastObject];
    NSLog(@"删除成功");

    [_lock unlock];
}
@end
//Log

2019-07-29 10:06:48.904648+0800 day16--线程安全[43603:4402260] 添加成功
2019-07-29 10:06:49.907641+0800 day16--线程安全[43603:4402259] 删除成功
复制代码

可以看到时间上差了1秒,正好是我们设定的sleep(1);。优点是可以让线程之间形成依赖,缺点是没有明确的条件。

NSConditionLock 可以实现线程依赖的锁

NSConditionLock是可以实现多个子线程进行线程间的依赖,A依赖于B执行完成,B依赖于C执行完毕则可以使用NSConditionLock来解决问题。 首先看下API

@property (readonly) NSInteger condition;//条件值
- (void)lockWhenCondition:(NSInteger)condition;//当con为condition进行锁住
//尝试加锁
- (BOOL)tryLock;
//当con为condition进行尝试锁住
- (BOOL)tryLockWhenCondition:(NSInteger)condition;
//当con为condition进行解锁
- (void)unlockWithCondition:(NSInteger)condition;
//NSDate 小余 limit进行 加锁
- (BOOL)lockBeforeDate:(NSDate *)limit;
//条件为condition 在limit之前进行加锁
- (BOOL)lockWhenCondition:(NSInteger)condition beforeDate:(NSDate *)limit;
复制代码

条件锁的使用,在lockWhenCondition:(NSInteger)condition的条件到达的时候才能进行正常的加锁和unlockWithCondition:(NSInteger)condition解锁,否则会阻塞线程。

- (void)otherTest{
    [[[NSThread alloc]initWithTarget:self selector:@selector(__test2) object:nil] start];
    [[[NSThread alloc]initWithTarget:self selector:@selector(__test1) object:nil] start];
    [[[NSThread alloc]initWithTarget:self selector:@selector(__test3) object:nil] start];

}
- (void)__test1{
    [_lock lockWhenCondition:1];
    NSLog(@"%s",__func__);
    [_lock unlockWithCondition:2];//解锁 并赋值2
}
- (void)__test2{
    [_lock lockWhenCondition:2];
    NSLog(@"%s",__func__);
    [_lock unlockWithCondition:3];//解锁 并赋值3
}
- (void)__test3{
    [_lock lockWhenCondition:3];
    NSLog(@"%s",__func__);
    [_lock unlockWithCondition:4];//解锁 并赋值4
}
@end
//log
-[FYCondLockDemo2 __test1]
-[FYCondLockDemo2 __test2]
-[FYCondLockDemo2 __test3]
复制代码

con = 1进行test1加锁和执行任务A,任务A执行完毕,进行解锁,并把值2赋值给lock,这是当con = 2的锁开始加锁,进入任务B,开始执行任务B,当任务B执行完毕,进行解锁并赋值为3,然后con=3的锁进行加锁,解锁并赋值4来进行线程之间的依赖。

dispatch_queue 特殊的锁

其实直接使用GCD的串行队列,也是可以实现线程同步的。串行队列其实就是线程的任务在队列中按照顺序执行,达到了锁的目的。

@interface FYSerialQueueDemo(){
    dispatch_queue_t _queue;
}@end
@implementation FYSerialQueueDemo
- (instancetype)init{
    if (self =[super init]) {
        _queue = dispatch_queue_create("fyserial.queue", DISPATCH_QUEUE_SERIAL);
    }
    return self;
}
- (void)__saleTicket{
    dispatch_sync(_queue, ^{
        [super __saleTicket];
    });
}
- (void)__getMonery{
    dispatch_sync(_queue, ^{
        [super __getMonery];
    });
}
- (void)__saveMonery{
    dispatch_sync(_queue, ^{
        [super __saveMonery];
    });
}
@end
//log
还剩 9 张票 - <NSThread: 0x600001211b40>{number = 3, name = (null)} 
还剩 8 张票 - <NSThread: 0x600001243700>{number = 4, name = (null)} 
还剩 7 张票 - <NSThread: 0x60000121dd80>{number = 5, name = (null)} 
还剩 6 张票 - <NSThread: 0x600001211b40>{number = 3, name = (null)} 
还剩 5 张票 - <NSThread: 0x600001243700>{number = 4, name = (null)} 
还剩 4 张票 - <NSThread: 0x60000121dd80>{number = 5, name = (null)} 
还剩 3 张票 - <NSThread: 0x600001211b40>{number = 3, name = (null)} 
还剩 2 张票 - <NSThread: 0x600001243700>{number = 4, name = (null)} 
还剩 1 张票 - <NSThread: 0x60000121dd80>{number = 5, name = (null)}
复制代码

dispatch_semaphore 信号量控制并发数量

当我们有大量任务需要并发执行,而且同时最大并发量为5个线程,这样子又该如何控制呢?dispatch_semaphore信号量正好可以满足我们的需求。dispatch_semaphore可以控制并发线程的数量,当设置为1时,可以作为同步锁来用,设置多个的时候,就是异步并发队列。

//初始化信号量 值为2,就是最多允许同时2个线程执行
_semaphore = dispatch_semaphore_create(2);
//生成多个线程进行并发访问test
- (void)otherTest{
    for (int i = 0; i < 10; i ++) {
        [[[NSThread alloc]initWithTarget:self selector:@selector(test) object:nil]start];
    }
}
- (void)test{
//如果信号量>0 ,让信号量-1,继续向下执行。
//如果信号量 <= 0;就会等待,等待时间是 DISPATCH_TIME_FOREVER
    dispatch_semaphore_wait(_semaphore, DISPATCH_TIME_FOREVER);
    sleep(2);//睡眠时间2s
    NSLog(@"%@",[NSThread currentThread]);
    //释放一个信号量
    dispatch_semaphore_signal(_semaphore);
}
//log

2019-07-29 11:17:53.233318+0800 day16--线程安全[47907:4529610] <NSThread: 0x600002c45240>{number = 4, name = (null)}
2019-07-29 11:17:53.233329+0800 day16--线程安全[47907:4529609] <NSThread: 0x600002c45200>{number = 3, name = (null)}
2019-07-29 11:17:55.233879+0800 day16--线程安全[47907:4529616] <NSThread: 0x600002c45540>{number = 10, name = (null)}
2019-07-29 11:17:55.233879+0800 day16--线程安全[47907:4529612] <NSThread: 0x600002c45440>{number = 6, name = (null)}
2019-07-29 11:17:57.238860+0800 day16--线程安全[47907:4529613] <NSThread: 0x600002c45480>{number = 7, name = (null)}
2019-07-29 11:17:57.238867+0800 day16--线程安全[47907:4529614] <NSThread: 0x600002c454c0>{number = 8, name = (null)}
2019-07-29 11:17:59.241352+0800 day16--线程安全[47907:4529615] <NSThread: 0x600002c45500>{number = 9, name = (null)}
2019-07-29 11:17:59.241324+0800 day16--线程安全[47907:4529611] <NSThread: 0x600002c45400>{number = 5, name = (null)}
2019-07-29 11:18:01.245790+0800 day16--线程安全[47907:4529618] <NSThread: 0x600002c455c0>{number = 12, name = (null)}
2019-07-29 11:18:01.245790+0800 day16--线程安全[47907:4529617] <NSThread: 0x600002c45580>{number = 11, name = (null)}
复制代码

一次最多2个线程同时执行任务,暂停时间是2s。 使用信号量实现线程最大并发锁, 同时只有2个线程执行的。

- (instancetype)init{
    if (self =[super init]) {
        _semaphore = dispatch_semaphore_create(1);
    }
    return self;
}
- (void)__saleTicket{
    dispatch_semaphore_wait(_semaphore, DISPATCH_TIME_FOREVER);
    [super __saleTicket];
    dispatch_semaphore_signal(_semaphore);
}
//log
还剩 9 张票 - <NSThread: 0x6000022e0c00>{number = 3, name = (null)} 
还剩 8 张票 - <NSThread: 0x6000022e0dc0>{number = 4, name = (null)} 
还剩 7 张票 - <NSThread: 0x6000022ce880>{number = 5, name = (null)} 
还剩 6 张票 - <NSThread: 0x6000022e0c00>{number = 3, name = (null)} 
还剩 5 张票 - <NSThread: 0x6000022e0dc0>{number = 4, name = (null)} 
还剩 4 张票 - <NSThread: 0x6000022ce880>{number = 5, name = (null)} 
还剩 3 张票 - <NSThread: 0x6000022e0c00>{number = 3, name = (null)} 
还剩 2 张票 - <NSThread: 0x6000022e0dc0>{number = 4, name = (null)} 
还剩 1 张票 - <NSThread: 0x6000022ce880>{number = 5, name = (null)} 
复制代码

@synchronized

@synchronized(id obj){}锁的是对象obj,使用该锁的时候,底层是对象计算出来的值作为key,生成一把锁,不同的资源的读写可以使用不同obj作为锁对象。

- (void)__saleTicket{
    @synchronized (self) {
        [super __saleTicket];
    }
 }
 //log
还剩 9 张票 - <NSThread: 0x60000057d5c0>{number = 3, name = (null)} 
还剩 8 张票 - <NSThread: 0x60000056f340>{number = 4, name = (null)} 
还剩 7 张票 - <NSThread: 0x60000057d500>{number = 5, name = (null)} 
还剩 6 张票 - <NSThread: 0x60000057d5c0>{number = 3, name = (null)} 
还剩 5 张票 - <NSThread: 0x60000056f340>{number = 4, name = (null)} 
还剩 4 张票 - <NSThread: 0x60000057d500>{number = 5, name = (null)} 
还剩 3 张票 - <NSThread: 0x60000057d5c0>{number = 3, name = (null)} 
还剩 2 张票 - <NSThread: 0x60000056f340>{number = 4, name = (null)} 
还剩 1 张票 - <NSThread: 0x60000057d500>{number = 5, name = (null)} 
复制代码

atmoic 原子操作

给属性添加atmoic修饰,可以保证属性的settergetter都是原子性操作,也就保证了settergetter的内部是线程同步的。 原子操作是最终调用了static inline void reallySetProperty(id self, SEL _cmd, id newValue, ptrdiff_t offset, bool atomic, bool copy, bool mutableCopy) objc-accessors.mm 48行,我们进入到函数内部

//设置属性原子操作
void objc_setProperty_atomic(id self, SEL _cmd, id newValue, ptrdiff_t offset)
{
    reallySetProperty(self, _cmd, newValue, offset, true, false, false);
}
//非原子操作设置属性
void objc_setProperty_nonatomic(id self, SEL _cmd, id newValue, ptrdiff_t offset)
{
    reallySetProperty(self, _cmd, newValue, offset, false, false, false);
}

static inline void reallySetProperty(id self, SEL _cmd, id newValue, ptrdiff_t offset, bool atomic, bool copy, bool mutableCopy)
{//偏移量等于0则是class指针
    if (offset == 0) {
        object_setClass(self, newValue);
        return;
    }
//其他的value
    id oldValue;
    id *slot = (id*) ((char*)self + offset);

    if (copy) {
    //如果是copy 用copyWithZone:
        newValue = [newValue copyWithZone:nil];
    } else if (mutableCopy) {
        //mutableCopy则调用mutableCopyWithZone:
        newValue = [newValue mutableCopyWithZone:nil];
    } else {
    //如果赋值和原来的相等 则不操作
        if (*slot == newValue) return;
        newValue = objc_retain(newValue);
    }

    if (!atomic) {//非原子操作 直接赋值
        oldValue = *slot;
        *slot = newValue;
    } else {//原子操作 加锁
    //锁和属性是一一对应的->自旋锁
        spinlock_t& slotlock = PropertyLocks[slot];
        slotlock.lock();
        oldValue = *slot;
        *slot = newValue;//赋值
        slotlock.unlock();//解锁
    }
    objc_release(oldValue);
}

id objc_getProperty(id self, SEL _cmd, ptrdiff_t offset, BOOL atomic) {
    if (offset == 0) {
        return object_getClass(self);
    }

    // Retain release world
    id *slot = (id*) ((char*)self + offset);
    if (!atomic) return *slot;//非原子操作 直接返回值

    // Atomic retain release world
    //原子操作 加锁->自旋锁
    spinlock_t& slotlock = PropertyLocks[slot];
    slotlock.lock();//加锁
    id value = objc_retain(*slot);
    slotlock.unlock();//解锁

    // for performance, we (safely) issue the autorelease OUTSIDE of the spinlock.
    return objc_autoreleaseReturnValue(value);
}

//以属性的地址为参数计算出key ,锁为value
StripedMap<spinlock_t> PropertyLocks;
复制代码

从源码了解到设置属性读取是self+属性的偏移量,当copymutableCopy会调用到[newValue copyWithZone:nil][newValue mutableCopyWithZone:nil],如果新旧值相等则不进行操作,非原子操作直接赋值,原子操作则获取spinlock_t& slotlock = PropertyLocks[slot]进行加锁、赋值、解锁操作。而且PropertyLocks是一个类,类有一个数组属性,使用*p计算出来的值作为key

我们提取出来关键代码

//原子操作 加锁
spinlock_t& slotlock = PropertyLocks[slot];
slotlock.lock();
oldValue = *slot;
*slot = newValue;//赋值
slotlock.unlock();//解锁
复制代码

使用自旋锁对赋值操作进行加锁,保证了setter()方法的安全性

//原子操作 加锁 ->自旋锁
spinlock_t& slotlock = PropertyLocks[slot];
slotlock.lock();//加锁
id value = objc_retain(*slot);
slotlock.unlock();//解锁
复制代码

取值之前进行加锁,取值之后进行解锁,保证了getter()方法的安全。

由上面得知atmoic仅仅是对方法setter()getter()安全,对成员变量不保证安全,对于属性的读写一般使用nonatomic,性能好,atomic读取频率高的时候会导致线程都在排队,浪费CPU时间。

大概使用者几种锁分别对卖票功能进行了性能测试, 性能分别1万次、100万次、1000万次锁花费的时间对比,单位是秒。(仅供参考,不同环境时间略有差异)

锁类型 1万次 100万次 1000万次
pthread_mutex_t 0.000309 0.027238 0.284714
os_unfair_lock 0.000274 0.028266 0.285685
OSSpinLock 0.030688 0.410067 0.437702
NSCondition 0.005067 0.323492 1.078636
NSLock 0.038692 0.151601 1.322062
NSRecursiveLock 0.007973 0.151601 1.673409
@synchronized 0.008953 0.640234 2.790291
NSConditionLock 0.229148 5.325272 10.681123
semaphore 0.094267 0.415351 24.699100
SerialQueue 0.213386 9.058581 50.820202

建议

平时我们简单使用的话没有很大的区别,还是推荐使用NSLock和信号量,最简单的是@synchronized,不用声明和初始化,直接拿来就用。

自旋锁、互斥锁比较

自旋锁和互斥锁各有优劣,代码执行频率高,CPU充足,可以使用互斥锁,频率低,代码复杂则需要互斥锁。

自旋锁

  • 自旋锁在等待时间比较短的时候比较合适
  • 临界区代码经常被调用,但竞争很少发生
  • CPU不紧张
  • 多核处理器

互斥锁

  • 预计线程等待时间比较长
  • 单核处理器
  • 临界区IO操作
  • 临界区代码比较多、复杂,或者循环量大
  • 临界区竞争非常激烈

锁的应用

简单读写锁

一个简单的读写锁,读写互斥即可,我们使用信号量,值设定为1.同时只能一个线程来操作文件,读写互斥。

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    self.semaphore = dispatch_semaphore_create(1);

    for (NSInteger i = 0; i < 10; i ++) {
        [[[NSThread alloc]initWithTarget:self selector:@selector(read) object:nil]start];
        [[[NSThread alloc]initWithTarget:self selector:@selector(write) object:nil]start];
    }
}

- (void)read{
    dispatch_semaphore_wait(self.semaphore, DISPATCH_TIME_FOREVER);
    NSLog(@"%s",__func__);
    dispatch_semaphore_signal(self.semaphore);
}
- (void)write{
    dispatch_semaphore_wait(self.semaphore, DISPATCH_TIME_FOREVER);
    NSLog(@"%s",__func__);
    dispatch_semaphore_signal(self.semaphore);
}
复制代码

当读写都是一个线程来操作,会降低性能,当多个线程在读资源的时候,其实不需要同步操作的,有读没写,理论上说不用限制异步数量,写入的时候不能读,才是真正限制线程性能的地方,读写锁具备以下特点

  1. 同一时间,只能有1个线程进行写操作
  2. 同一时间,允许有多个线程进行读的操作
  3. 同一时间,不允许读写操作同时进行

典型的多读单写,经常用于文件等数据的读写操作,我们实现2种

读写锁 pthread_rwlock

这是有c语言封装的读写锁

//初始化读写锁
int pthread_rwlock_init(pthread_rwlock_t * __restrict,
        const pthread_rwlockattr_t * _Nullable __restrict)
//读上锁
pthread_rwlock_rdlock(pthread_rwlock_t *)
//尝试加锁读
pthread_rwlock_tryrdlock(pthread_rwlock_t *)
//尝试加锁写
int pthread_rwlock_trywrlock(pthread_rwlock_t *)
//写入加锁
pthread_rwlock_wrlock(pthread_rwlock_t *)
//解锁
pthread_rwlock_unlock(pthread_rwlock_t *)
//销毁锁属性
pthread_rwlockattr_destroy(pthread_rwlockattr_t *)
//销毁锁
pthread_rwlock_destroy(pthread_rwlock_t * )
复制代码

pthread_rwlock_t使用很简单,只需要在读之前使用pthread_rwlock_rdlock,读完解锁pthread_rwlock_unlock,写入前需要加锁pthread_rwlock_wrlock,写入完成之后解锁pthread_rwlock_unlock,任务都执行完了可以选择销毁pthread_rwlock_destroy或者等待下次使用。

@property (nonatomic,assign) pthread_rwlock_t rwlock;

- (void)dealloc{
    pthread_rwlock_destroy(&_rwlock);//销毁锁
}
//初始化读写锁
pthread_rwlock_init(&_rwlock, NULL);

dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
    for (NSInteger i = 0; i < 5; i ++) {
        dispatch_async(queue, ^{
            [[[NSThread alloc]initWithTarget:self selector:@selector(readPthreadRWLock) object:nil]start];
            [[[NSThread alloc]initWithTarget:self selector:@selector(writePthreadRWLock) object:nil]start];
        });
    }

- (void)readPthreadRWLock{
    pthread_rwlock_rdlock(&_rwlock);
    NSLog(@"读文件");
    sleep(1);
    pthread_rwlock_unlock(&_rwlock);
}
- (void)writePthreadRWLock{
    pthread_rwlock_wrlock(&_rwlock);
    NSLog(@" 写入文件");
    sleep(1);
    pthread_rwlock_unlock(&_rwlock);
}

//log
2019-07-30 10:47:16 读文件
2019-07-30 10:47:16 读文件
2019-07-30 10:47:17 写入文件
2019-07-30 10:47:18 写入文件
2019-07-30 10:47:19 读文件
2019-07-30 10:47:19 读文件
2019-07-30 10:47:19 读文件
2019-07-30 10:47:20 写入文件
2019-07-30 10:47:21 写入文件
2019-07-30 10:47:22 写入文件
复制代码

读文件会出现同一秒读多次,写文件同一秒只有一个。

异步栅栏调用 dispatch_barrier_async

栅栏大家都见过,为了分开一个地区而使用的,线程的栅栏函数是分开任务的执行顺序

操作 任务 任务 任务
A B
A B
C
C
A
A B

这个函数传入的并发队列必须是通过dispatch_queue_create创建,如果传入的是一个串行的或者全局并发队列,这个函数便等同于dispatch_async的效果。

//初始化 异步队列
self.rwqueue = dispatch_queue_create("rw.thread", DISPATCH_QUEUE_CONCURRENT);
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
for (NSInteger i = 0; i < 5; i ++) {
    dispatch_async(queue, ^{
        [self readBarryier];
        [self readBarryier];
        [self readBarryier];
        [self writeBarrier];
    });
}

- (void)readBarryier{
//添加任务到rwqueue
    dispatch_async(self.rwqueue, ^{
        NSLog(@"读文件 %@",[NSThread currentThread]);
        sleep(1);
    });
}
- (void)writeBarrier{
//barrier_async添加任务到self.rwqueue中
    dispatch_barrier_async(self.rwqueue, ^{
        NSLog(@"写入文件 %@",[NSThread currentThread]);
        sleep(1);
    });
}

//log

2019-07-30 11:16:53 读文件 <NSThread: 0x600001ae0740>{number = 9, name = (null)}
2019-07-30 11:16:53 读文件 <NSThread: 0x600001ae8500>{number = 10, name = (null)}
2019-07-30 11:16:53 读文件 <NSThread: 0x600001ae8040>{number = 8, name = (null)}
2019-07-30 11:16:53 读文件 <NSThread: 0x600001ac3a80>{number = 11, name = (null)}
2019-07-30 11:16:54 写入文件<NSThread: 0x600001ac3a80>{number = 11, name = (null)}
2019-07-30 11:16:55 写入文件<NSThread: 0x600001ac3a80>{number = 11, name = (null)}
2019-07-30 11:16:56 写入文件<NSThread: 0x600001ac3a80>{number = 11, name = (null)}
复制代码

读文件会出现同一秒读多个,写文件同一秒只有一个。

读写任务都添加到异步队列rwqueue中,使用栅栏函数dispatch_barrier_async拦截一下,实现读写互斥,读可以异步无限读,写只能一个同步写的功能。

总结

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