【Linux C/C++ 线程同步 】Linux API 读写锁的编程使用

简介: 【Linux C/C++ 线程同步 】Linux API 读写锁的编程使用

读写锁介绍

读写锁比mutex有更高的适用性,可以多个线程同时占用读模式的读写锁,但是只能一个线程占用写模式的读写锁。

  • 当读写锁是写加锁状态时,在这个锁被解锁之前,所有试图对这个锁加锁的线程都会被阻塞;
  • 当读写锁在读加锁状态时,所有试图以读模式对它进行加锁的线程都可以得到访问权,但是以写模式对它进行枷锁的线程将阻塞;
  • 当读写锁在读模式锁状态时,如果有另外线程试图以写模式加锁,读写锁通常会阻塞随后的读模式锁请求,这样可以避免读模式锁长期占用,而等待的写模式锁请求长期阻塞;

读写锁使用场景

读写锁适用对数据结构进行读的次数比写的次数多的情况下,因为可以进行读锁共享。而互斥锁通常用于读写串行。

读写锁相关接口函数

  • 初始化读写锁
int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock, const pthread_rwlockattr_t *restrict attr);
//使用互斥变量之前必须初始化

参数:

rwlock:读写锁

attr:为NULL时,表示默认属性

返回值:

    若成功,返回0

    若失败,返回出错编号

  • 销毁读写锁
int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);
//在释放他们底层的内存之前必须摧毁

返回值:

    若成功,返回0

    若失败,返回出错编号

  • 读加锁和写加锁
int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);
 
int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);
 
int pthread_rwlock_unlock(pthread_rwlock_t *rwlock);

返回值:

    若成功,返回0

    若失败,返回出错编号

  • 非阻塞获得读锁和写锁
int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock);
 
int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock);

返回值:

    若成功,返回0

    若失败,返回出错编号

  • 带有超时的读写锁
int pthread_rwlock_timedwrlock(pthread_rwlock_t *restrict rwlock,const struct timespec *restrict abs_timeout);
 
int pthread_rwlock_timedrdlock(pthread_rwlock_t *restrict rwlock,const struct timespec *restrict abs_timeout);
//使应用程序获取读写锁时避免永久阻塞状态。

参数:

abs_timeout:指向timespec结构,指定线程应该停止阻塞的时间。

返回值:

    若成功,返回0

    若失败,返回出错编号

  • 初始化/销毁读写锁属性
int pthread_rwlockattr_init(pthread_rwlockattr_t *attr);
//将使用实现定义的所有属性的默认值初始化读写锁属性对象attr。
//如果指定已初始化的attr属性对象,则结果未定义。
 
int pthread_rwlockattr_destroy(pthread_rwlockattr_t *attr);
//销毁读写锁属性对象。

在读写锁属性对象用于初始化一个或多个读写锁之后,影响属性对象(包括破坏)的任何函数都不会影响任何先前初始化的读写锁。

返回值:

    若成功,返回0

    若失败,返回出错编号

  • 获取/设置读写锁属性对象的进程共享属性
int pthread_rwlockattr_getpshared(const pthread_rwlockattr_t *restrict attr, int *restrict pshared);
//从attr引用的初始化属性对象中获取进程共享属性的值。
//将attr的进程共享属性的值存储到由pshared参数引用的对象中。
 
int pthread_rwlockattr_setpshared(pthread_rwlockattr_t *attr,int pshared);
//在attr引用的初始化属性对象中设置进程共享属性。

参数:

进程共享属性应设置为 PTHREAD_PROCESS_SHARED 以允许任何可访问读写锁定分配的内存的线程对读写锁操作,即使读写锁分配在内存中这是由多个进程共享的。

如果进程共享属性为 PTHREAD_PROCESS_PRIVATE 则读写锁只能在与初始化读写锁的线程相同的进程内创建的线程进行操作;

如果不同进程的线程尝试对这样的读写锁进行操作,则行为是未定义的。进程共享属性的默认值应为 PTHREAD_PROCESS_PRIVATE

返回值:

    成功 ,返回 0

    失败 ,返回 出错编号

使用示例

  • 读写锁简单使用代码段
//创建读写锁 
pthread_rwlock_t lock;
pthread_rwlock_init(&lock,NULL);
int data;
/** 写端线程  **/
     //加写锁
    pthread_rwlock_wrlock(&lock);
    data++;
    printf("==write:%lu,%d\n",pthread_self(),data);
    //解锁
    pthread_rwlock_unlock(&lock);
    sleep(1);
/** 读端线程     **/
    //加读锁
    pthread_rwlock_rdlock(&lock);
    printf("==read:%lu,%d\n",pthread_self(),data);
    pthread_rwlock_unlock(&lock);
    sleep(1);
//释放读写锁资源
pthread_rwlock_destroy(&lock);


目录
相关文章
|
3月前
|
存储 Linux API
【Linux进程概念】—— 操作系统中的“生命体”,计算机里的“多线程”
在计算机系统的底层架构中,操作系统肩负着资源管理与任务调度的重任。当我们启动各类应用程序时,其背后复杂的运作机制便悄然展开。程序,作为静态的指令集合,如何在系统中实现动态执行?本文带你一探究竟!
【Linux进程概念】—— 操作系统中的“生命体”,计算机里的“多线程”
|
1月前
|
并行计算 Linux
Linux内核中的线程和进程实现详解
了解进程和线程如何工作,可以帮助我们更好地编写程序,充分利用多核CPU,实现并行计算,提高系统的响应速度和计算效能。记住,适当平衡进程和线程的使用,既要拥有独立空间的'兄弟',也需要在'家庭'中分享和并行的成员。对于这个世界,现在,你应该有一个全新的认识。
146 67
|
3月前
|
Linux
Linux编程: 在业务线程中注册和处理Linux信号
本文详细介绍了如何在Linux中通过在业务线程中注册和处理信号。我们讨论了信号的基本概念,并通过完整的代码示例展示了在业务线程中注册和处理信号的方法。通过正确地使用信号处理机制,可以提高程序的健壮性和响应能力。希望本文能帮助您更好地理解和应用Linux信号处理,提高开发效率和代码质量。
77 17
|
3月前
|
Linux
Linux编程: 在业务线程中注册和处理Linux信号
通过本文,您可以了解如何在业务线程中注册和处理Linux信号。正确处理信号可以提高程序的健壮性和稳定性。希望这些内容能帮助您更好地理解和应用Linux信号处理机制。
74 26
|
3月前
|
消息中间件 Linux C++
c++ linux通过实现独立进程之间的通信和传递字符串 demo
的进程间通信机制,适用于父子进程之间的数据传输。希望本文能帮助您更好地理解和应用Linux管道,提升开发效率。 在实际开发中,除了管道,还可以根据具体需求选择消息队列、共享内存、套接字等其他进程间通信方
91 16
|
6月前
|
Java 开发者
Java多线程编程中的常见误区与最佳实践####
本文深入剖析了Java多线程编程中开发者常遇到的几个典型误区,如对`start()`与`run()`方法的混淆使用、忽视线程安全问题、错误处理未同步的共享变量等,并针对这些问题提出了具体的解决方案和最佳实践。通过实例代码对比,直观展示了正确与错误的实现方式,旨在帮助读者构建更加健壮、高效的多线程应用程序。 ####
|
5月前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
409 2
|
6月前
|
设计模式 Java 开发者
Java多线程编程的陷阱与解决方案####
本文深入探讨了Java多线程编程中常见的问题及其解决策略。通过分析竞态条件、死锁、活锁等典型场景,并结合代码示例和实用技巧,帮助开发者有效避免这些陷阱,提升并发程序的稳定性和性能。 ####
|
6月前
|
缓存 Java 开发者
Java多线程编程的陷阱与最佳实践####
本文深入探讨了Java多线程编程中常见的陷阱,如竞态条件、死锁和内存一致性错误,并提供了实用的避免策略。通过分析典型错误案例,本文旨在帮助开发者更好地理解和掌握多线程环境下的编程技巧,从而提升并发程序的稳定性和性能。 ####
|
5月前
|
缓存 Java 调度
多线程编程核心:上下文切换深度解析
在现代计算机系统中,多线程编程已成为提高程序性能和响应速度的关键技术。然而,多线程编程中一个不可避免的概念就是上下文切换(Context Switching)。本文将深入探讨上下文切换的概念、原因、影响以及优化策略,帮助你在工作和学习中深入理解这一技术干货。
101 10