Linux:使用读写锁使线程同步

基础与控制原语

读写锁

与互斥量类似,但读写锁允许更高的并行性。其特性为:写独占,读共享。

读写锁状态:

一把读写锁具备三种状态:

1. 读模式下加锁状态 (读锁)

2. 写模式下加锁状态 (写锁)

3. 不加锁状态

读写锁特性:    

  1. 读写锁是"写模式加锁"时, 解锁前,所有对该锁加锁的线程都会被阻塞。
  2. 读写锁是"读模式加锁"时, 如果线程以读模式对其加锁会成功;如果线程以写模式加锁会阻塞。
  3. 读写锁是"读模式加锁"时, 既有试图以写模式加锁的线程,也有试图以读模式加锁的线程。那么读写锁会阻塞随后的读模式锁请求。优先满足写模式锁。读锁、写锁并行阻塞,写锁优先级高

读写锁也叫共享-独占锁。当读写锁以读模式锁住时,它是以共享模式锁住的;当它以写模式锁住时,它是以独占模式锁住的。写独占、读共享。

读写锁非常适合于对数据结构读的次数远大于写的情况。

主要应用函数:

pthread_rwlock_init函数

pthread_rwlock_destroy函数

pthread_rwlock_rdlock函数

pthread_rwlock_wrlock函数

pthread_rwlock_tryrdlock函数

pthread_rwlock_trywrlock函数

pthread_rwlock_unlock函数

以上7 个函数的返回值都是:成功返回0, 失败直接返回错误号。

pthread_rwlock_t类型    用于定义一个读写锁变量。

pthread_rwlock_t rwlock;

pthread_rwlock_init函数

初始化一把读写锁

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

参2:attr表读写锁属性,通常使用默认属性,传NULL即可。

pthread_rwlock_destroy函数

销毁一把读写锁

int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);

pthread_rwlock_rdlock函数

以读方式请求读写锁。(常简称为:请求读锁)

int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);

pthread_rwlock_wrlock函数

以写方式请求读写锁。(常简称为:请求写锁)

int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);

pthread_rwlock_unlock函数

解锁

int pthread_rwlock_unlock(pthread_rwlock_t *rwlock);

pthread_rwlock_tryrdlock函数

非阻塞以读方式请求读写锁(非阻塞请求读锁)

int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock);

pthread_rwlock_trywrlock函数

非阻塞以写方式请求读写锁(非阻塞请求写锁)

int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock);

读写锁示例

看如下示例,同时有多个线程对同一全局数据读、写操作。

#include
<stdio.h>

#include
<stdlib.h>

#include
<pthread.h>

#include
<unistd.h>

pthread_rwlock_t rwlock;

long
int love;

void *pth_wr(void *arg)

{

int i = (int)arg;

while (love <= 520)

{

pthread_rwlock_wrlock(&rwlock);//请求写锁

printf("write================全局变量love = %ld, 我是%d号线程。\n", love += 40, i + 1);

pthread_rwlock_unlock(&rwlock);//解锁

sleep(1);

}

return
NULL;

}

void *pth_rd(void *arg)

{

int i = (int)arg;

while (love <= 520)

{

pthread_rwlock_rdlock(&rwlock);//请求读锁

printf("全局变量love = %ld, 我是%d号线程。————— - read\n", love, i + 1);

pthread_rwlock_unlock(&rwlock);//解锁

sleep(1);

}

return
NULL;

}

int main(void)

{

pthread_t pth[10];

int i;

pthread_rwlock_init(&rwlock, NULL);

for (i = 0; i != 5; i++)//写

{

pthread_create(&pth[i], NULL, pth_wr, (void *)i);

}

for (i = 0; i != 5; i++)//读

{

pthread_create(&pth[5 + i], NULL, pth_rd, (void *)i);

}

while (1)

{

if (love >= 520)

{

for (int j = 0; j != 10; j++)

{

pthread_cancel(pth[j]);//杀死线程

pthread_join(pth[j], NULL);//回收线程

}

break;

}

}

return 0;

}

在里面,定义了一个全局变量love,当love大于520的时候就停止读写并杀死回收线程。读写锁加解锁的位置和互斥量加解锁的位置一样,直接看结果吧:看,读数据的时候时一致的,并且读出来的数据与前面写进去的数据保持一致,并没有出现读出的数据和写入的数据不一致的情况。虽然线程执行顺序没有顺序,但是这并不重要,这只是内核调度和线程争夺资源的结果,我们关心的不是线程执行顺序,而是结果。很明显,加了锁之后整整齐齐,是我们想要的效果;当然这样写在64位上编译的时候会有警告:原因我在上一篇博客上说了的,这里不再做解释。

现在我们来看看不加锁的情况,代码就是把那些锁去掉就是。

这是每次加40的结果;

这是每次加20的结果,可以看到,次数越多,混乱的情况就越严重。

原文地址:https://www.cnblogs.com/love-DanDan/p/8723931.html

时间: 2024-10-12 06:10:03

Linux:使用读写锁使线程同步的相关文章

线程同步——用户模式下线程同步——Slim读写锁实现线程同步

1 //Slim读/写锁实现线程同步 2 SRWlock 的目的和关键段相同:对同一资源进行保护,不让其它线程访问. 3 但是,与关键段不同的是,SRWlock允许我们区分哪些想要读取资源的线程(读取者线程) 4 和哪些想要更新资源值的线程(写入者线程).让所有读取者资源在同一时刻访问共享资源应该是 5 可行的,这是因为仅仅读取资源并不存在破坏数据的风险.只有当写入者线程想要对资源进行更新时才需要同步. 6 这种情况下,写入者线程应该独占资源访问权:任何线程,无论是读取还是写入者线程,都不许访问

使用读写锁实现线程同步

简介: 读写锁与互斥量类似,但读写锁允许更高的并行性.其特性为:写独占,读共享. 读写锁特性: 读写锁是"写模式加锁"时,解锁前,所有对该锁加锁的线程都会被阻塞. 读写锁是"读模式加锁"时,如果线程以读模式对其加锁会成功.如果线程以写模式加锁会阻塞. 读写锁是"读模式加锁"时,如果有另外线程试图以写模式加锁,读写锁通常会阻塞随后的读模式锁请求,这样可以避免读模式锁长期占用,而等待的写模式锁请求长期阻塞: 读写锁非常适合于对数据结构读的次数远大于写

java读写锁实现数据同步访问

锁机制最大的改进之一就是ReadWriteLock接口和它的唯一实现类ReentrantReadWriteLock.这个类有两个锁,一个是读操作锁,另一个是写操作锁.使用读操作锁时可以允许多个线程同时访问,但是使用写操作锁时只允许一个线程进行.在一个线程执行写操作时,其他线程不能够执行读操作. 下面我们将通过范例学习如何使用ReadWriteLock接口编写程序.这个范例将使用ReadWriteLock接口控制对价格对象的访问,价格对象存储了两个产品的价格. 1. 创建一个价格信息类Prices

【Linux】Mutex互斥量线程同步的例子

0.互斥量  Windows下的互斥量 是个内核对象,每次WaitForSingleObject和ReleaseMutex时都会检查当前线程ID和占有互斥量的线程ID是否一致. 当多次Wait**时就要对应多次ReleaseMutex, 当ReleaseMutex过多次数时如果发现当前占有互斥量的线程ID和当前调用ReleaseMutex的线程ID不一致时仅仅返回FLASE,GetLastError返回ERROR_NOT_OWNER,没有其他副作用. 当占有mutex的线程在Release之前退

笔记2 linux多线程 读写锁

//read write lock #include<stdio.h> #include<unistd.h> #include<pthread.h> struct test { char a[10]; char b[10]; char c[10]; }yb = {"111","222","33333"}; static int j=0; pthread_rwlock_t mutex_1; void Print1

Linux同步技术之读写锁

互斥锁试图将想进入临界区的所有线程都阻塞住,但是有时候该临界区会涉及由这些线程共享的一个或多个数据的访问或更新,这时候我们就需要用到读写锁. 系统读写锁的分配规则: (1)只要有没有线程持有给定的读写锁用于写,那么任意数量的线程可以持有该读写锁用于读.(系统规定写锁优先,但是可以更改为读锁优先) (2)仅当没有线程持有某个读写锁用于读或用于写时,才能分配该读写锁用于写. 读写锁用于读称为共享锁,读写锁用于写称为独占锁. 读写锁的获取与释放: int pthread_rwlock_rdlock(p

浅析线程间通信二:读写锁和自旋锁

上文讨论了互斥量和条件变量用于线程的同步,本文将讨论读写锁和自旋锁的使用,并给出了相应的代码和注意事项,相关代码也可在我的github上下载. 读写锁 对于互斥量要么是锁住状态要么是不加锁锁状态,而且一次只有一个线程可以对其加锁,而读写锁对线程的读数据加锁请求和写数据加锁请求进行了区分,从而在某些情况下,程序有更高的并发性.对于读写锁,一次只有一个线程可以占有写模式的读写锁,但是多个线程可以同时占有读模式的读写锁.虽然读写锁的实现各不相同,但当读写锁处于读模式锁住状态时,如果有另外的线程试图以写

11.6 线程同步

11.6.1 互斥Example11.6.2 避免死锁Example11.6.3 pthread_mutex_timedlock 函数Example11.6.4Reader-Writer LocksExample11.6.5 带有超时功能的读写锁11.6.6 条件变量Example11.6.7 自旋锁11.6.8 BarriersExample 当多个线程控制流需要共享内存的时候,我们需要确保每一个线程所看到的数据是一致的.如果一个线程使用别的线程不会读取或者修改的数据,那么一致性问题并不会出现

信号量、互斥锁,读写锁和条件变量的区别

信号量强调的是线程(或进程)间的同步:“信号量用在多线程多任务同步的,一个线程完成了某一个动作就通过信号量告诉别的线程,别的线程再进行某些动作(大家都 在sem_wait的时候,就阻塞在那里).当信号量为单值信号量是,也可以完成一个资源的互斥访问.有名信号量:可以用于不同进程间或多线程间的互斥与同步 创建打开有名信号量 sem_t *sem_open(const char *name, int oflag); sem_t *sem_open(const char *name, int oflag