1、前言
在前面我们已经讲过了iOS中的线程同步技术,主要通过加锁实现。
对于读写操作,一般都比较耗时耗性能,为了保持其安全性,我们一般要采取“多读单写”模式:
(1)同一时间,只能有1个线程进行写的操作
(2)同一时间,允许有多个线程进行读的操作
(3)同一时间,不允许既有写的操作,又有读的操作
iOS中提供了以下两种方法去实现:
pthread_rwlock 读写锁
dispatch_barrier_async 异步栅栏调用
2、pthread_rwlock和dispatch_barrier_async的用法
pthread_rwlock读写锁
//初始化读写锁
pthread_rwlock_t rwlock;
pthread_rwlock_init(&rwlock, NULL);
//读 - 加锁
pthread_rwlock_rdlock(&rwlock);
//读 - 尝试加锁
pthread_rwlock_tryrdlock(&rwlock);
//写 - 加锁
pthread_rwlock_wrlock(&rwlock);
//写 - 尝试加锁
pthread_rwlock_trywrlock(&rwlock);
//解锁
pthread_rwlock_unlock(&rwlock);
//销毁锁
pthread_rwlock_destroy(&rwlock);
这个例子就是读写操作下的线程保护操作:pthread_rwlock读写锁的运用
============================================================
#import "ViewController.h"
#import <pthread.h>
@interface ViewController ()
@property(nonatomic,assign)pthread_rwlock_t rwLock;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
//初始化_rwLock
pthread_rwlock_init(&_rwLock, NULL);
for (int i=0; i<5; i++) {
[[[NSThread alloc] initWithTarget:self selector:@selector(read) object:nil] start];
[[[NSThread alloc] initWithTarget:self selector:@selector(write) object:nil] start];
}
}
-(void)read{
pthread_rwlock_rdlock(&_rwLock);
sleep(1);
NSLog(@"%s",__func__);
pthread_rwlock_unlock(&_rwLock);
}
-(void)write{
pthread_rwlock_wrlock(&_rwLock);
sleep(1);
NSLog(@"%s",__func__);
pthread_rwlock_unlock(&_rwLock);
}
-(void)dealloc{
pthread_rwlock_destroy(&_rwLock);
}
@end
dispatch_barrier_async异步栅栏调用
dispatch_barrier_async函数传入的并发队列必须是自己通过dispatch_queue_cretate创建的,如果传入的是一个串行或是一个全局的并发队列,那这个函数便等同于dispatch_async函数的效果。
//初始化队列
dispatch_queue_t queue = dispatch_queue_create("rw_queue", DISPATCH_QUEUE_CONCURRENT);
//读
dispatch_async(queue, ^{
});
//写
dispatch_barrier_async(queue, ^{
});
#import "ViewController.h"
@interface ViewController ()
@property(nonatomic,strong)dispatch_queue_t queue;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
self.queue = dispatch_queue_create("rwQueue", DISPATCH_QUEUE_CONCURRENT);
for (int i=0; i<5; i++) {
[self read];
[self write];
}
}
-(void)read{
dispatch_async(self.queue, ^{
sleep(1);
NSLog(@"%s",__func__);
});
}
-(void)write{
dispatch_barrier_async(self.queue, ^{
sleep(1);
NSLog(@"%s",__func__);
});
}
@end
3、扩展
nonatomic 和 atomic
atomic:原子,不可分割
atomic用于保证属性setter、getter的原子性操作,相当于在getter和setter内部加了线程同步的锁
但是它并不能保证使用属性的过程是线程安全的
网友评论