Linux系统编程多线程之读写锁讲解

一.前言

1.当有一个线程已经持有互斥锁时,互斥锁将所有试图进入临界区的线程都阻塞住。但是考虑一种情形,

当前持有互斥锁的线程只是要读访问共享资源,而同时有其它几个线程也想读取这个共享资源,但是由于互斥锁的排它性,

所有其它线程都无法获取锁,也就无法读访问共享资源了,但是实际上多个线程同时读访问共享资源并不会导致问题.
2.在对数据的读写操作中,更多的是读操作,写操作较少,例如对数据库数据的读写应用为了满足当前能够允许多个读出,

但只允许一个写入的需求,线程提供了读写锁来实现。
3.读写锁的特点:

1.如果有其它线程读数据,则允许其它线程执行读操作,但不允许写操作。

2.如果有其它线程写数据,则其它线程都不允许读、写操作。

3.写是独占的,写的优先级高
读写锁的常见函数:

int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock,const pthread_rwlockattr_t*restrict attr);

int pthread_rwlock_destroy(pthread_rwlock_t*rwlock);

int pthread_rwlock_rdlock(pthread_rwlock_t*rwlock);

int pthread_rwlock_tryrdlock(pthread_rwlock_t*rwlock);

int pthread_rwlock_wrlock(pthread_rwlock_t*rwlock);

int pthread_rwlock_trywrlock(pthread_rwlock_t*rwlock);

int pthread_rwlock_unlock(pthread_rwlock_t*rwlock);

二.实例代码

复制代码
/*
    int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock,const pthread_rwlockattr_t*restrict attr);

    int pthread_rwlock_destroy(pthread_rwlock_t*rwlock);
    
    int pthread_rwlock_rdlock(pthread_rwlock_t*rwlock);
    
    int pthread_rwlock_tryrdlock(pthread_rwlock_t*rwlock);
    
    int pthread_rwlock_wrlock(pthread_rwlock_t*rwlock);
    
    int pthread_rwlock_trywrlock(pthread_rwlock_t*rwlock);
    
    int pthread_rwlock_unlock(pthread_rwlock_t*rwlock);
*/

/*
案例:8个线程操作同一个全局变量
      3个线程不定时写这个全局变量,5个线程不定时读这个全局变量
*/
#include <stdio.h>
#include <unistd.h>
#include <pthread.h>

//创建一个共享资源
int num = 1;
pthread_mutex_t mutex;
pthread_rwlock_t rwlock;
void * writeNum(void *arg)
{
    while(1)
    {
        // pthread_mutex_lock(&mutex);
        pthread_rwlock_wrlock(&rwlock);
        num++;
        printf("++write,tid:%ld,num:%d\n",pthread_self(),num);
        pthread_rwlock_unlock(&rwlock);
        usleep(100);
    }
    return NULL;
}
void * readNum(void *arg)
{
    while(1)
    {
        // pthread_mutex_lock(&mutex);
        pthread_rwlock_rdlock(&rwlock);
        printf("=====read,tid:%ld,num:%d\n",pthread_self(),num);
        // pthread_mutex_unlock(&mutex);
        pthread_rwlock_unlock(&rwlock);
        usleep(100);
    }
    return NULL;
}
int main()
{
    pthread_mutex_init(&mutex,NULL);
    pthread_rwlock_init(&rwlock,NULL);
    pthread_t wtids[3],rtids[5];
    for(int i = 0; i < 3; i++)
    {
        pthread_create(&wtids[i],NULL,writeNum,NULL);
    }
    for(int i = 0; i < 5; i++)
    {
        pthread_create(&rtids[i],NULL,readNum,NULL);
    }
    //设置线程分离
    for(int i = 0; i < 3; i++)
    {
        pthread_detach(wtids[i]);
    }
    for(int i = 0; i < 5; i++)
    {
        pthread_detach(rtids[i]);
    }

    pthread_exit(NULL);
    pthread_mutex_destroy(&mutex);
    pthread_rwlock_destroy(&rwlock);
    return 0;
}
相关推荐
是娇娇公主~26 分钟前
C++ 中 std::deque 的原理?它内部是如何实现的?
开发语言·c++·stl
SuperEugene1 小时前
Axios 接口请求规范实战:请求参数 / 响应处理 / 异常兜底,避坑中后台 API 调用混乱|API 与异步请求规范篇
开发语言·前端·javascript·vue.js·前端框架·axios
xuxie992 小时前
N11 ARM-irq
java·开发语言
路溪非溪2 小时前
Linux下蓝牙框架的数据流
linux·arm开发·驱动开发
能不能别报错2 小时前
openclaw-linux部署教程+mimo-v2-pro
linux·运维·服务器
wefly20172 小时前
从使用到原理,深度解析m3u8live.cn—— 基于 HLS.js 的 M3U8 在线播放器实现
java·开发语言·前端·javascript·ecmascript·php·m3u8
luanma1509803 小时前
PHP vs C++:编程语言终极对决
开发语言·c++·php
寂静or沉默3 小时前
2026最新Java岗位从P5-P7的成长面试进阶资源分享!
java·开发语言·面试
济6173 小时前
ARM Linux 驱动开发篇---基于 pinctrl+GPIO 子系统的蜂鸣器驱动开发(设备树版)--- Ubuntu20.04
linux·嵌入式·嵌入式linux驱动开发
AiGuoHou13 小时前
Debian/Ubuntu 各个版本一键更换国内镜像源
linux·ubuntu·国内源·debian·镜像源·换源