• iOS读写安全方案


    思考如下实现以下场景?

    • 同一时间,只能有1个线程进行写的操作
    • 同一时间,允许多个线程进行读的操作
    • 同一时间,不允许既有写的操作,又有读的操作

    上面的场景就是典型的“多读单写”,经常用于文件等数据的读写操作,iOS的实现方案有

    1. pthread_rwlock:读写锁
    2. dispatch_barrier_async:异步栅栏调用

    1.  pthread_rwlock 互斥锁(会休眠)

    1.1  知识讲解

    1.2  demo实例讲解

    #import "ViewController.h"
    #import <pthread.h>
    
    @interface ViewController ()
    @property (assign, nonatomic) pthread_rwlock_t lock;
    @end
    
    @implementation ViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
        
        // 初始化锁
        pthread_rwlock_init(&_lock, NULL);
        
        dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
        
        for (int i = 0; i < 10; i++) {
            dispatch_async(queue, ^{
                [self read];
            });
            dispatch_async(queue, ^{
                [self write];
            });
        }
    }
    
    
    - (void)read {
        //读锁
        pthread_rwlock_rdlock(&_lock);
        
        sleep(1);
        NSLog(@"%s", __func__);
        
        pthread_rwlock_unlock(&_lock);
    }
    
    - (void)write
    {
        //写锁
        pthread_rwlock_wrlock(&_lock);
        
        sleep(1);
        NSLog(@"%s", __func__);
        
        pthread_rwlock_unlock(&_lock);
    }
    
    // pthread_rwlock_destroy 是pthread锁,要销毁
    - (void)dealloc
    {
        pthread_rwlock_destroy(&_lock);
    }

     

    1.3  执行结果

      

    2.  dispatch_barrier_async 栅栏

    2.1  知识讲解

    这个函数传入的并发队列 必须是自己通过dispatch_queue_create创建的

    如果传入的是一个串行队列或者是全局的并发队列,那这个函数便等同于dispatch_async 函数的效果,起不到栅栏的作用

     

    2.2  demo实例讲解

    #import "ViewController.h"
    #import <pthread.h>
    
    @interface ViewController ()
    @property (strong, nonatomic) dispatch_queue_t queue;
    @end
    
    @implementation ViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
        
        self.queue = dispatch_queue_create("rw_queue", DISPATCH_QUEUE_CONCURRENT);
        
        for (int i = 0; i < 10; i++) {
            dispatch_async(self.queue, ^{
                [self read];
            });
            
            dispatch_async(self.queue, ^{
                [self read];
            });
            
            dispatch_async(self.queue, ^{
                [self read];
            });
            
            dispatch_barrier_async(self.queue, ^{
                [self write];
            });
        }
    }
    
    
    - (void)read {
        sleep(1);
        NSLog(@"read");
    }
    
    - (void)write
    {
        sleep(1);
        NSLog(@"write");
    }
    
    @end

     

    2.3  执行结果

    上面就是两种常用的iOS 读写安全操作,欢迎指正!!!

  • 相关阅读:
    parted 2T以上磁盘分区和挂载
    MySQL 表空间恢复
    MySQL 绿色版安装
    EF core
    让vs自动提示没有using的类
    Android Studio 的 Gradle 面板没有 Task
    JMS微服务远程调用性能测试 vs .Net Core gRPC服务
    electron打包,使用electron-packager
    EasyNetQ(RabbitMQ)在处理消息时,如果抛出异常,继续不断发送到订阅队列,不断处理(也就是不自动确认消息已到达)
    调用系统默认浏览器,打开网页
  • 原文地址:https://www.cnblogs.com/guohai-stronger/p/12906137.html
Copyright © 2020-2023  润新知