美文网首页
多线程 之 多线程的读写安全

多线程 之 多线程的读写安全

作者: ychen3022 | 来源:发表于2018-12-24 21:08 被阅读23次
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内部加了线程同步的锁
但是它并不能保证使用属性的过程是线程安全的

相关文章

  • 多线程 之 多线程的读写安全

    1、前言 在前面我们已经讲过了iOS中的线程同步技术,主要通过加锁实现。对于读写操作,一般都比较耗时耗性能,为了保...

  • 多线程-读写安全

    多线程31-读写安全01-简介 文件操作(IO操作)读取文件往文件中写入内容不能允许读取和写入同时进行我们之前做的...

  • 多线程-读写安全

  • 多线程读写安全

    多读单写 要想实现多线程的读写安全,必须满足“读读”并发,“读写”、“写写”互斥。即: 同一时间,只能有1个线程进...

  • go sync包的读写锁RWMutex的使用

    sync包的读写锁RWMutex的使用(sync.RWMutex) 我们使用“读写”锁的场景主要是在多线程的安全操...

  • JAVA面试汇总(二)多线程(六)

    JAVA多线程内容比较多,今天写完了第六篇,后边还有七(肯定最后一篇了)。 如何保证多线程读写文件的安全?(1)读...

  • atomic 和 nonatomic 的区别

    atomic 提供多线程安全,防止读写未完成的时候被另外一个线程读写,造成数据错误nontomic 在自己管理内存...

  • iOS多线程读写崩溃分析

    最近再次遇到多线程读写导致的crash 问题,写了一个测试demo,记录分析过程。 上面是暴力重现多线程读写的崩溃...

  • Redis各版本特性

    Redis6.0 多线程IO Redis 6引入多线程IO,但多线程部分只是用来处理网络数据的读写和协议解析,执行...

  • iOS多线程到底不安全在哪里?

    iOS多线程到底不安全在哪里? iOS多线程到底不安全在哪里?

网友评论

      本文标题:多线程 之 多线程的读写安全

      本文链接:https://www.haomeiwen.com/subject/ycaakqtx.html