1块资源可能会被多个线程共享,也就是多个线程可能会访问同一块资源
比如多个线程访问同一个对象、同一个变量、同一个文件
当多个线程访问同一块资源时,很容易引发数据错乱和数据安全问题
OC在定义属性时有nonatomic和atomic两种选择
原子和非原子属性的选择
atomic:原子属性,线程安全,需要消耗大量的资源 ,为setter方法加锁(默认就是atomic)
nonatomic:非原子属性,不会为setter方法加锁,非线程安全,适合内存小的移动设备
互斥锁使用格式
@synchronized(锁对象) { // 需要锁定的代码 }
注意:锁定1份代码只用1把锁,用多把锁是无效的
指令@synchronized()需要一个参数。该参数可以使任何的Objective-C对象,包括self。这个对象就是互斥信号量。他能够让一个线程对一段代码进行保护,避免别的线程执行该段代码。针对程序中的不同的关键代码段,我们应该分别使用不同的信号量。只有在应用程序编程执行多线程之前就创建好所有需要的互斥信号量对象来避免线程间的竞争才是最安全的。
@synchronized 的作用是创建一个互斥锁,保证此时没有其它线程对self对象进行修改。这个是objective-c的一个锁定令牌,防止self对象在同一时间内被其它线程访问,起到线程的保护作用。 一般在公用变量的时候使用,如单例模式或者操作类的可以用来创建单例:
static LaoSiJiManager3 *_singleton = nil;
@implementation LaoSiJiManager3
- (instancetype)sharedManager_Sync {
if (_singleton == nil) {
// 上锁,确保同一时间只有一人开锁
@synchronized (self) {
if (_singleton == nil) {
_singleton = [[self alloc] init];
}
}
}
return _singleton;
}
static变量中使用。
@synchronized(self) {
do something;
}
atomic 加锁原理
@property (assign, atomic) int age;
- (void)setAge:(int)age
{
@synchronized(self) {
_age = age;
}
}
NSCache是线程安全的,在多线程操作中,不需要对Cache加锁。
通俗讲:两个售票员售票,未加互斥锁时两个人可能卖的是同一张票,加了之后,等一个人把票卖了,另一个人才能继续卖;
互斥锁的优缺点
优点:能有效防止因多线程抢夺资源造成的数据安全问题
缺点:需要消耗大量的CPU资源
互斥锁的使用前提:多条线程抢夺同一块资源
相关专业术语:线程同步,多条线程按顺序地执行任务
互斥锁,就是使用了线程同步技术