服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - IOS - iOS中读写锁的简单实现方法实例

iOS中读写锁的简单实现方法实例

2021-12-29 14:45头疼脑胀的代码搬运工 IOS

读写锁是计算机程序的并发控制的一种同步机制,也称“共享-互斥锁”、多读者-单写者锁,读操作可并发重入,写操作是互斥的,这篇文章主要给大家介绍了关于iOS中读写锁的简单实现方法,需要的朋友可以参考下

 

废话开篇

iOS 下的多线程的技术的应用衍生出了锁的机制,试想,如果 iOS 下没有多线程的概念,所有的代码都会在同步环境下执行,那么,也就不会产生争夺资源情况的发生,当然,也就没有办法利用多核的优势。所以,多线程的应用是广布的,而锁的应用是局部的,所以,二者应相辅相成,来达到提高运行效率的同时提高程序运行的稳定性。

 

思考一、对于锁的类型的理解

基本的三种锁的类型:互斥锁、自旋锁、读写锁。

其中,互斥锁 多线程在访问加锁中的临界区前,会进入休眠,一直等待解锁后系统调度
;自旋锁 多线程在访问加锁中的临界区前,不进入休眠,会一直忙等。 读写锁 是一种思想,本质就是利用 互斥锁 来实现特定的应用场景:多读并行、读与写互斥,写与写互斥;对于其他的类型的锁,比如:信号量、条件锁、递归锁,可以理解为是由以上基本类型的锁实现的上层封装。

 

思考二、读写锁的实现逻辑

如果有这样一块公共资源,它的写入是比较耗时,那么,在这段时间内要避免程序再次的进行写入操作和读取操作,这样可以避免产生争夺资源的问题,当然,读取的过程可以并行。

先上一段代码,这里模拟一个比较耗时的写入过程,在模拟一个快速读取的过程。

//读写锁
- (void)readAndWriteLock
{
    //写
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
            NSLog(@"我开始写");
            for (int i = 0; i < 10000; i++) {
            }
            NSLog(@"我写完了");

    });

    //读
    for (int i = 0; i < 10; i++) {
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
            NSLog(@"我开始读%d",i);
            for (int j = 0; j < i; j++) {
            }
            NSLog(@"我读完了%d",i);
        });
    }
}

在写入的异步内执行了循环 10000 次的操作来模拟耗时任务;后面开辟了多线程进行读取,在读取的异步里,实现简单的不同数量的循环来模拟耗时任务,这里的耗时次数远小于写入。

打印如下

iOS中读写锁的简单实现方法实例

通过输出可以清晰的看到,写入过程中还有很多读取的操作在同时进行。致于是先开始读取还是开始写入,这里其实并不用去关心,本身它们也是由系统决定的,但是,这里需要控制一下代码,来实现读写互斥,即在写的过程中,禁止读取操作的介入。同时,也需要实现写写互斥。对于,多读,其实这里并不需要过多干涉,因为,本身就允许多读的逻辑。

 

思考三、简单封装读写锁,满足读写逻辑

利用互斥锁封装读取加、解锁;写入加、解锁

//初始化读取锁
static pthread_mutex_t r_plock =  PTHREAD_MUTEX_INITIALIZER;
//初始化写入锁
static pthread_mutex_t w_plock =  PTHREAD_MUTEX_INITIALIZER;
//记录当前读取次数,因为只要其值不为0,那么,就说明程序在读取操作,这里停止写入操作
static int current_read_times = 0;

进行读取加锁

//读加锁
- (void)readLock
{
    pthread_mutex_lock(&r_plock);
    current_read_times ++;
    if (current_read_times == 1) {
        pthread_mutex_lock(&w_plock);
    }
    pthread_mutex_unlock(&r_plock);
}

这里首先进行 读取锁 加锁,这里加锁的目的并不是锁定读取过程,而是锁定了修改 current_read_times 的过程,当 current_read_times 变更后,如果为 1,那么,就对 写入锁 加锁,这个写入锁就是锁住写入过程的。这两个锁的应用部分是有区别的。

进行读取解锁

//读解锁
- (void)readUnLock
{
    pthread_mutex_lock(&r_plock);
    current_read_times --;
    if (current_read_times == 0) {
        pthread_mutex_unlock(&w_plock);
    }
    pthread_mutex_unlock(&r_plock);
}

这里首先进行 读取锁 加锁,目的还是对 current_read_times 修改的锁,可以总结一下,读写锁本质并不对多读进行限制,所以,这里的读取锁是锁住 current_read_times 修改过程,在加锁的情况下进行 写入锁 状态的变更,实现 读与写的互斥。后面进行状态判断,如果 current_read_times 为 0,说明当前所以读取完成了,那么,对 写入锁 进行解锁,解锁后写入操作就可以正常进行。这里解锁的判断为 == 0 与 加锁的判断 == 1 是一对,这样就满足了,一个加锁过程对应一个解锁条件。不会出现只有加锁后而没有解锁的情况。

进行写入加锁

//写加锁
- (void)writeLock
{
    pthread_mutex_lock(&w_plock);
}

这里仅仅是对 写入 操作进行加锁。

进行写入解锁

//写解锁
- (void)writeUnLock
{
    pthread_mutex_unlock(&w_plock);
}

这里仅仅是对 写入 操作进行解锁。

最后的代码

//读写锁
- (void)readAndWriteLock
{
    //写
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
            [self writeLock];
            NSLog(@"我开始写");
            for (int i = 0; i < 10000; i++) {

            }
            NSLog(@"我写完了");
            [self writeUnLock];
    });
    
    //读
    for (int i = 0; i < 10; i++) {
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
            [self readLock];
            NSLog(@"我开始读%d",i);
            for (int j = 0; j < i; j++) {

            }
            NSLog(@"我读完了%d",i);
            [self readUnLock];
        });
    }
}

打印如下

iOS中读写锁的简单实现方法实例

可以看待写入过程是完整的一个打印顺序,而读取过程由于没有锁的保护并没有按顺序执行。

iOS中读写锁的简单实现方法实例

一个简单的 读写锁 就完成了。代码拙劣,大神勿笑。

 

总结

到此这篇关于iOS中读写锁的简单实现的文章就介绍到这了,更多相关iOS读写锁实现内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://juejin.cn/post/7025421259231985677

延伸 · 阅读

精彩推荐
  • IOS关于iOS自适应cell行高的那些事儿

    关于iOS自适应cell行高的那些事儿

    这篇文章主要给大家介绍了关于iOS自适应cell行高的那些事儿,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的...

    daisy6092021-05-17
  • IOSiOS通过逆向理解Block的内存模型

    iOS通过逆向理解Block的内存模型

    自从对 iOS 的逆向初窥门径后,我也经常通过它来分析一些比较大的应用,参考一下这些应用中某些功能的实现。这个探索的过程乐趣多多,不仅能满足自...

    Swiftyper12832021-03-03
  • IOS解析iOS开发中的FirstResponder第一响应对象

    解析iOS开发中的FirstResponder第一响应对象

    这篇文章主要介绍了解析iOS开发中的FirstResponder第一响应对象,包括View的FirstResponder的释放问题,需要的朋友可以参考下...

    一片枫叶4662020-12-25
  • IOSiOS中tableview 两级cell的展开与收回的示例代码

    iOS中tableview 两级cell的展开与收回的示例代码

    本篇文章主要介绍了iOS中tableview 两级cell的展开与收回的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    J_Kang3862021-04-22
  • IOSiOS布局渲染之UIView方法的调用时机详解

    iOS布局渲染之UIView方法的调用时机详解

    在你刚开始开发 iOS 应用时,最难避免或者是调试的就是和布局相关的问题,下面这篇文章主要给大家介绍了关于iOS布局渲染之UIView方法调用时机的相关资料...

    windtersharp7642021-05-04
  • IOSIOS开发之字典转字符串的实例详解

    IOS开发之字典转字符串的实例详解

    这篇文章主要介绍了IOS开发之字典转字符串的实例详解的相关资料,希望通过本文能帮助到大家,让大家掌握这样的方法,需要的朋友可以参考下...

    苦练内功5832021-04-01
  • IOSIOS 屏幕适配方案实现缩放window的示例代码

    IOS 屏幕适配方案实现缩放window的示例代码

    这篇文章主要介绍了IOS 屏幕适配方案实现缩放window的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要...

    xiari5772021-06-01
  • IOSiOS 雷达效果实例详解

    iOS 雷达效果实例详解

    这篇文章主要介绍了iOS 雷达效果实例详解的相关资料,需要的朋友可以参考下...

    SimpleWorld11022021-01-28