rwlock 使用介绍

本文介绍了读写锁(rwlock)的概念,它允许多个读取者同时访问共享资源,但只允许一个写入者。文章详细阐述了读写锁的基本函数,包括初始化、获取读锁、尝试读锁、获取写锁、尝试写锁、解锁和销毁读写锁。并提供了一个多线程读写操作的C语言代码示例,展示了读写锁在并发控制中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


一、rwlock 介绍

当一个线程持有互斥锁时,互斥锁将所有试图进入临界区的线程都阻塞住。但是考虑一种情形,当前持有互斥锁的线程只是要读访问共享资源,而同时其它线程也想读这个共享资源,但是由于互斥锁的排它性,所有其它线程都无法获取锁,也就无法读访问共享资源了,但实际上多个线程同时读访问共享资源并不会导致问题
为了满足可以多个读权限一个写权限,提供了读写锁来实现,规则如下:

  • 如果某线程申请了读锁,其它线程可以再申请读锁,但不能申请写锁
  • 如果某线程申请了写锁,其它线程不能申请读锁,也不能申请写锁

二、rwlock 基本函数

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

可使用宏静态初始化读写锁,如下:等价于用NULL指定attr调用 pthread_rwlock_init 函数

pthread_rwlock_t  rwlock = PTHREAD_RWLOCK_INITIALIZER;

2. int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock );

以阻塞方式在读写锁上获取读锁。如果没有写者持有该锁,并且没有写者阻塞在该锁上,则调用线程会获取读锁。如果调用线程未获取读锁,则它将阻塞直到它获取了该锁
一个线程可以在一个读写锁上多次执行读锁定。线程可以成功调用 pthread_rwlock_rdlock() 函数 n 次,但是之后该线程必须调用 pthread_rwlock_unlock() 函数 n 次才能解除锁定

3. int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock);

以非阻塞的方式来在读写锁上获取读锁。如果有任何的写者持有该锁或有写者阻塞在该读写锁上,则立即返回失败

4. int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock );

在读写锁上获取写锁。如果没有写者持有该锁,并且没有读者持有该锁,则调用线程会获取写锁。如果调用线程未获取写锁,则它将阻塞直到它获取了该锁

5. int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock);

以非阻塞的方式来在读写锁上获取写锁。如果有任何的读者或写者持有该锁,则立即返回失败

6. int pthread_rwlock_unlock (pthread_rwlock_t *rwlock);

无论是读锁或写锁,都可以通过此函数解锁

7. int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);

用于销毁读写锁,释放所有相关联的资源(由 pthread_rwlock_init 自动申请的资源)

三、rwlock 代码示例

#include<stdio.h>
#include<unistd.h>
#include<pthread.h>

pthread_rwlock_t rwlock;
int num = 1;

void *read_1(void *arg) // 读操作,其他线程允许读操作,却不允许写操作
{
    while(1)
    {
        pthread_rwlock_rdlock(&rwlock);
        printf("read_1 num=%d\n", num);
        pthread_rwlock_unlock(&rwlock);
        sleep(1);
    }
}

void *read_2(void *arg) // 读操作,其他线程允许读操作,却不允许写操作
{
    while(1)
    {
        pthread_rwlock_rdlock(&rwlock);
        printf("read_2 num=%d\n", num);
        pthread_rwlock_unlock(&rwlock);
        sleep(2);
    }
}


void *write_1(void *arg) // 写操作,其它线程都不允许读或写操作
{
    while(1)
    {
        pthread_rwlock_wrlock(&rwlock);
        num++;
        printf("write_1 num=%d\n", num);
        pthread_rwlock_unlock(&rwlock);
        sleep(2);
    }
}

void *write_2(void *arg) // 写操作,其它线程都不允许读或写操作
{
    while(1)
    {
        pthread_rwlock_wrlock(&rwlock);
        num++;
        printf("write_2 num=%d\n", num);
        pthread_rwlock_unlock(&rwlock);
        sleep(1);
    }
}

int main()
{
    pthread_t ptd1, ptd2, ptd3, ptd4;

    pthread_rwlock_init(&rwlock, NULL);
    pthread_create(&ptd1, NULL, read_1, NULL);
    pthread_create(&ptd2, NULL, read_2, NULL);
    pthread_create(&ptd3, NULL, write_1, NULL);
    pthread_create(&ptd4, NULL, write_2, NULL);
    pthread_join(ptd1, NULL);
    pthread_join(ptd2, NULL);
    pthread_join(ptd3, NULL);
    pthread_join(ptd4, NULL);
    pthread_rwlock_destroy(&rwlock);
    return 0;
}

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值