Linux多线程基础(5):读写锁(rwlock大白话讲解)

前文中已经介绍了互斥锁用于解决数据共享带来的问题,为什么又要引入读写锁呢?别慌,我将用最通俗的解释来帮助大家理解!

1.为什么要引入读写锁

假设我们对一个线程使用了互斥锁,一旦上锁,其他线程对共享数据既不能读取,也不能写入.聪明的你可能会发现:如果大部分线程仅仅想读取共享数据,这一上互斥锁,都得等该线程完成并解锁后才可以读取数据,效率可谓是大大降低了呀!

为了解决上述面临的问题,读写锁诞生了!

2.读写锁的定义

读写锁(pthread_rwlock_t)是一种线程同步机制,允许多个线程对同一共享资源进行读操作,但同一时间只允许有一个线程对共享数据进行写操作。

读写锁共有两种操作:pthread_rwlock_rdlock(读锁)pthread_rwlock_wrlock(写锁)

2.1 pthread_rwlock_rdlock(读锁)

读锁是用来给读操作加锁的。当一个进程给共享资源加上读锁后,其他线程仍然可以给共享资源加读锁,并可以成功读取数据,但是不能加写锁.读锁的定义如下:

int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);

其中参数rwlock为指向pthread_rwlock_t类型的指针。若函数成功执行,返回0,失败返回错误码.

举个例子:我先讲解一个这段代码的主要意思,代码中一共有两个线程,线程1线程2都加了读锁,目的是看看在加了读锁后是否会出现线程1线程2交替进行读操作的情况.

#include <pthread.h>  
#include<string.h>  
#include<stdio.h>  
#include<unistd.h>  
#include<stdlib.h>  
#include<iostream>  
int number=10;//全局变量
pthread_rwlock_t rwlock; 
void *pth1(void* str){
    pthread_rwlock_rdlock(&rwlock);//加读锁
       for(int i=0;i<10;i++){
        std::cout<<"线程1加读锁输出结果:"<<number<<std::endl;
        usleep(10);
    } 
    pthread_rwlock_unlock(&rwlock);   //解读锁
    pthread_exit(0);
}
void *pth2(void* str){
    pthread_rwlock_rdlock(&rwlock);//加读锁
     for(int i=0;i<10;i++){
        std::cout<<"线程2加读锁输出结果:"<<number<<std::endl;  //检测是否可以读取到数据
     }
    usleep(5);
    pthread_rwlock_unlock(&rwlock);   //解读锁
    pthread_exit(0); 
}
int main(void){
    pthread_t thread1,thread2;
    pthread_rwlock_init(&rwlock,NULL);  //读写锁初始化
    pthread_create(&thread1,NULL,pth1,NULL);
    pthread_create(&thread2,NULL,pth2,NULL);
    pthread_join(thread1,NULL);
    pthread_join(thread2,NULL);
    pthread_rwlock_destroy(&rwlock);  //读写锁销毁
    std::cout<<"主函数退出"<<std::endl;
}  

 运行程序得到下面的结果:

可以看出, 线程1线程2出现了交替进行读操作的情况.说明读写锁允许多个线程对同一共享资源进行读操作.

2.2pthread_rwlock_wrlock(写锁)

写锁是用来给写操作加锁的。当一个线程给共享资源加上写锁后,其他线程既不可加读锁也不能加写锁,其实就相当于是一个互斥锁.写锁的定义如下:

int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);

其中参数rwlock为指向pthread_rwlock_t类型的指针。若函数成功执行,返回0,失败返回错误码.

举个例子:我先讲解一个这段代码的主要意思,代码中一共有两个线程,线程1加了写锁,线程2加了读锁,目的是看看在加了写锁的情况下是否可以进行读操作.

#include <pthread.h>  
#include<string.h>  
#include<stdio.h>  
#include<unistd.h>  
#include<stdlib.h>  
#include<iostream>  
int number;//全局变量
pthread_rwlock_t rwlock; 
void *pth1(void* str){
    pthread_rwlock_wrlock(&rwlock);//加写锁
       for(int i=0;i<20;i++){
        number++;
        std::cout<<"线程1加写锁输出结果:"<<number<<std::endl;
        usleep(10);
    } 
    pthread_rwlock_unlock(&rwlock);   //解读锁
    pthread_exit(0);
}
void *pth2(void* str){
    pthread_rwlock_rdlock(&rwlock);//加读锁
    std::cout<<"线程2加读锁输出结果:"<<number<<std::endl;  //检测是否可以读取到数据

    usleep(5);
    pthread_rwlock_unlock(&rwlock);   //解读锁
    pthread_exit(0); 
}
int main(void){
    pthread_t thread1,thread2;
    pthread_rwlock_init(&rwlock,NULL);  //读写锁初始化
    pthread_create(&thread1,NULL,pth1,NULL);
    pthread_create(&thread2,NULL,pth2,NULL);
    pthread_join(thread1,NULL);
    pthread_join(thread2,NULL);
    pthread_rwlock_destroy(&rwlock);  //读写锁销毁
    std::cout<<"主函数退出"<<std::endl;
}  

运行程序得到下面的结果:

 可以看出,只有当线程1完成所有任务,并对写锁解锁后,线程2才能进行读操作!

3.总结

当一个多线程程序中,读操作比较多时,我们这时就使用读写锁,写操作比较多时,就用互斥锁即可.

  • 10
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Redisson的是一种特殊的机制,它允许多个线程同时占有模式的,但一次只能有一个线程占有模式的。当处于模式时,所有试图对该线程都会被阻塞,直到被解。当处于模式时,所有试图以模式对其加线程都可以获得访问权,但如果线程希望以模式对此进行加,它必须等待所有的线程释放。通常情况下,当处于模式定状态时,如果有其他线程试图以模式加通常会阻塞随后的模式请求,以避免模式长期占用,而等待的模式请求长期阻塞。适用于对数据结构的次数远远多于次数的情况,因为模式定时可以共享,而模式定时是独占的。[3] 在Redisson中,可以通过以下方式使用: 1. 获取RedissonClient对象。 2. 使用RedissonClient对象的getReadWriteLock方法获取对象。 3. 使用对象的readLock方法获取对象,或使用writeLock方法获取对象。 4. 在需要加的代码块中,使用对象的lock方法进行加,使用对象的lock方法进行加。 5. 在加的代码块执行完毕后,使用对象的unlock方法进行解,使用对象的unlock方法进行解。 例如,在验证的请求中,可以使用RedissonClient对象的getReadWriteLock方法获取对象,然后使用对象的lock方法进行加取数据后再使用unlock方法进行解。在验证的请求中,可以使用相同的方式获取对象,然后使用对象的lock方法进行加入数据后再使用unlock方法进行解。[2] 总之,Redisson的是一种灵活且高效的机制,可以在多线程环境下实现对共享资源的安全访问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值