在程序开发过程中,经常使用到线程,多线程操作数据时。难免发生一些不可控的情况,造成数据不安全。这个时候我们就用了锁,常用的锁有哪些?有os_unfair_lock、NSLock、NSCondition、NSRecursiveLock
等。
一. 线程锁
1、原子属性和锁
通过一个小例子看下无锁情况下的demo:
@property (nonatomic ,assign) int count;
- (void)test {
for (int i = 0; i < 10; i ++) {
dispatch_async(dispatch_get_global_queue(0, 0), ^{
self.count --;
NSLog(@"%d",self.count);
});
}
}
复制代码
无锁的情况下打印是无序的。哪我们把 (atomic ,assign) int count;
改成原子属性呢? 看到打印结果也是无序的。atomic 原子属性也无法保证数据安全
。
哪我们进入objc源码看看:
set方法里没有锁的操作
get方法里有锁的操作
看到源码有加锁的操作,哪为什么数据还是不安全呢。self.count --;
这个操作即有get操作
,也有set操作
。但是锁只在get中,所以我们打印的结果并不是有序的。 上图是各种锁的加锁10000次的性能。
锁分:互斥锁(闲等),自旋锁(忙等),读写锁3大类。
ios中有:OSSpinLock、dispatch_semaphore、pthread_mutex、NSCondition、NSLock、pthread_mutex(recursive)、NSRecursiveLock、NSConditionLock、@synchronized
9把锁。
2、OSSpinLock,os_unfair_lock锁
OSSpinLock
:(它有一个bug,线程优先级反转的问题:优先级低的线程资先执行,然后其他优先级高的在执行会一直占用CPU。优先级底底那个线程会资源无法释放。)
- iOS10之后被移除。
OS_SPINLOCK_INIT
初始化锁 。OSSpinLockLock(&spinlock)
加锁,参数为OSSPINLOCK地址。OSSpinLockUnlock(&spinlock)
解锁,参数是OSSpinLock地址。OSSpinLockTry(&spinlock)
尝试上锁,参数是OSSpinLock地址。如果返回false,表示上锁失败,锁正在被其他线程持有。如果返回true,表示上锁成功。
os_unfair_lock
:
iOS10之后开始支持
,用于取代OSSpinLock。OS_UNFAIR_LOCK_INIT
初始化锁。os_unfair_lock_lock
加锁。参数为os_unfair_lock地址。os_unfair_lock_unlock
解锁。参数为os_unfair_lock地址。os_unfair_lock_trylock
尝试加锁。参数为os_unfair_lock地址。如果成功返回true。如果锁已经被锁定则返回false。os_unfair_lock_assert_owner
参数为os_unfair_lock地址。如果当前线程未持有指定的锁或者锁已经被解锁,则触发崩溃。os_unfair_lock_assert_not_owner
参数为os_unfair_lock地址。如果当前线程持有指定的锁,则触发崩溃。
os_unfair_lock例子:
@property (nonatomic ,assign) os_unfair_lock unfairLock;
self.unfairLock = OS_UNFAIR_LOCK_INIT;
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
for (int i = 0; i<10; i++) {
dispatch_async(dispatch_get_global_queue(0, 0), ^{
[self unfairLock_test];
});
}
}
-(void)unfairLock_test {
os_unfair_lock_lock(&_unfairLock);//加锁
self.count --;
NSL