【Linux多线程编程】6. 线程锁(3)——条件变量
admin
2024-05-08 10:37:33
0

前言

上篇文章【Linux多线程编程】5. 线程锁(2)——死锁、读写锁介绍了几种能够引发死锁的场景,并介绍了何为读写锁,以及读写锁的使用场景。本文首要先描述读写锁的基本使用,然后介绍第三种线程同步方式——条件变量。

读写锁的使用

  1. 创建读写锁
    pthread_rwlock_t rwlock;
    
  2. 初始化读写锁
    初始化读写锁的方法有两种,一种是直接将 PTHREAD_RWLOCK_INITIALIZER 宏赋值给读写锁变量,一种是通过函数初始化
    pthread_rwlock_t myRWLock = PTHREAD_RWLOCK_INITIALIZER; // 方法1,直接赋值
    int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock, const pthread_rwlockattr_t *restrict attr);
    // 方法2:函数调用 
    // 参数:
    // rwlock 参数用于指定要初始化的读写锁变量;
    // attr 参数用于自定义读写锁变量的属性,置为 NULL 时表示以默认属性初始化读写锁。
    // 一般 attr 参数传NULL
    
  3. 释放读写锁资源
    int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);
    
  4. 加锁
    int pthread_rwlock_rdlock(pthread_rwlock_t* rwlock); // 加读锁
    int pthread_rwlock_wrlock(pthread_rwlock_t* rwlock); // 加写锁
    int pthread_rwlock_tryrdlock(pthread_rwlock_t* rwlock); 
    int pthread_rwlock_trywrlock(pthread_rwlock_t* rwlock); 
    
    pthread_rwlock_tryxx 与 pthread_rwlock_rdlock/pthread_rwlock_wrlock 的区别在于:

    若此时无法加对应的锁,则 pthread_rwlock_rdlock/pthread_rwlock_wrlock 会阻塞等待锁被释放
    pthread_rwlock_tryxx 则是直接返回EBUSY 错误码,不会阻塞线程。

  5. 解锁
    int pthread_rwlock_unlock(pthread_rwlock_t* rwlock);
    
  6. 锁的销毁
    int pthread_rwlock_destroy(pthread_rwlock_t* rwlock);
    

读写锁代码示例

创建3个读线程,1个写线程,读线程输出临界资源 x 的值,写线程对临界资源 x 的值 + 1,并输出写后的值。
因此,读线程的工作函数 read_thread 调用 pthread_rwlock_rdlock加读锁。
写线程的工作函数 write_thread 调用 pthread_rwlock_wrlock 加写锁。

// thread.c
#include 
#include 
#include 
int x = 0; // 临界资源 x
//创建读写锁变量
pthread_rwlock_t myrwlock;// 读线程,只加读锁即可
void* read_thread(void* args){printf("------%u read_thread ready\n", pthread_self()); // pthread_self() 输出当前线程的线程IDwhile (1) {sleep(1);//请求读锁pthread_rwlock_rdlock(&myrwlock);printf("read_thread: %u,x=%d\n", pthread_self(), x);sleep(1);//释放读写锁pthread_rwlock_unlock(&myrwlock);}return NULL;
}// 写线程,需要加写锁
void* write_thread(void* param)
{printf("------%u write_thread ready!\n",pthread_self());while (1) {sleep(1);// 请求写锁pthread_rwlock_wrlock(&myrwlock);++x; // 修改临界资源 x 的值,需要加写锁printf("write_thread: %u,x=%d\n", pthread_self(), x);sleep(1);//释放读写锁pthread_rwlock_unlock(&myrwlock);}return NULL;
}int main()
{int i;//初始化读写锁pthread_rwlock_init(&myrwlock, NULL);//创建 3 个读 x 变量的线程pthread_t readThread[3];for (i = 0; i < 3; ++i) {pthread_create(&readThread[i], NULL, read_thread, NULL);}//创建 1 个修改 x 变量的线程pthread_t writeThread;pthread_create(&writeThread, NULL, write_thread, NULL);//等待各个线程执行完成pthread_join(writeThread, NULL);for (int i = 0; i < 3; ++i) {pthread_join(readThread[i], NULL);}//销毁读写锁pthread_rwlock_destroy(&myrwlock);return 0;
}

编译生成可执行文件
gcc thread.c -o thread -lpthread
执行 ./thread 后结果如下:

------1134741248 read_thread ready
------1113761536 read_thread ready
------1103271680 write_thread ready!
------1124251392 read_thread ready
read_thread: 1124251392,x=0
read_thread: 1113761536,x=0
read_thread: 1134741248,x=0
write_thread: 1103271680,x=1
read_thread: 1134741248,x=1
read_thread: 1124251392,x=1
read_thread: 1113761536,x=1
write_thread: 1103271680,x=2
read_thread: 1124251392,x=2
read_thread: 1113761536,x=2
read_thread: 1134741248,x=2

可以看出线程 1124251392、1113761536、1134741248为读线程,每次读取 x 的值;
线程 1103271680 为写线程,每次对 x 值 + 1;写线程写完 x 后,读线程再读取 x 的值就会比上次读取的值多1。

条件变量

条件变量是一种特殊的线程同步的方式,但是与之前我们介绍的加锁保证临界资源的安全访问不同,条件变量是用来阻塞并唤醒线程的,需要与互斥锁配合才能真正达到线程同步的效果。
锁体现的是一种竞争,我离开了,通知你进来。而条件变量体现的是一种协作,我准备好了,通知你开始吧。
条件变量通常与互斥锁使用,以多线程执行业务,业务执行完后需要退出主程序的例子举例,语义可以描述为:

主线程:我已经创建了业务线程,接下来需要等待 stop = 1,stop = 1 时请唤醒我,我来继续执行后面的退出操作。
条件变量(总指挥):好的,我会一直循环观测 stop 的值,当它变为 1 时,我唤醒主线程;小弟1,请先请求互斥锁配合,获取当前stop的值,并阻塞主线程等待。
条件变量(小弟1):好的;互斥锁,请协助我,先获取到 stop 的锁,我需要获取当前的 stop 值,我将一直执行阻塞主线程操作,直到被唤醒;别担心,我拿到你给我的锁后,内部会释放这把锁,然后再阻塞主线程,因此不会因为我加了 stop 锁导致其他线程阻塞。
互斥锁:收到;现在没有其他线程要 stop 锁,这把锁可以给你。(给条件变量(小弟1)锁)。
条件变量(小弟1):主线程阻塞已完成,内部已释放 stop 锁,我将在这里等待被唤醒。条件变量(总指挥),请在需要的时候唤醒我。
条件变量(总指挥):收到,我已设置了 stop = 1 时的唤醒程序,届时将有线程唤醒条件变量(小弟1)。业务线程s,如果觉得可以退出了,请执行我预设的唤醒程序。
业务线程s:我们正在执行一些业务程序,请求已收到,请放心,我们业务线程之间会协商最终的业务完毕时间,并由指定的某个业务线程执行你预设的唤醒程序。

上面用一大段对话模拟了条件变量和互斥锁的配合,看起来写了很多内容,其实代码很简单,先贴出来压压惊:

pthread_mutex_t stop_lock;
pthread_cond_t stop_cond;
unsigned stop;
void wait_stop() 
{pthread_mutex_lock(&stop_lock); // 加锁while (stop == 0) { // 循环的作用后续文章介绍pthread_cond_wait(&stop_cond, &stop_lock); // 阻塞调用 wait_stop 的线程,等待被 pthread_cond_signal 唤醒}stop= stop-1;pthread_mutex_unlock(&stop_lock);
}void signal_stop() 
{pthread_mutex_lock(&stop_lock); if (stop == 0) {pthread_cond_signal(&stop_cond);  // 某线程调用 signal_stop ,唤醒线程}stop = 1; // 设置 stop 标志pthread_mutex_unlock(&stop_lock);
}

关于条件变量的详细使用将在下篇文章介绍,请及时关注。

相关内容

热门资讯

蔚来法务部:恶意诋毁蔚来,多个... 1月19日消息,蔚来法务部发布公告称,“妄图网络”等多个网络账号的使用人及运营方毛祥鉴、长沙县星沙小...
亚太主要股指收盘,韩国综合指数... 1月19日消息,亚太主要股指收盘,韩国综合指数涨1.32%报4904.66点,再创收盘历史新高;日经...
婴童概念尾盘异动,汉商集团直线... 1月19日消息,尾盘婴童概念异动拉升,汉商集团直线涨停,康芝药业、戴维医疗、孩子王、共同药业、孩子王...
A股收评:沪指涨0.29%报4... 1月19日消息,A股三大指数今日涨跌不一,截至收盘,沪指涨0.29%报4114点,深证成指涨0.09...
可扩展中性原子阵列技术获验证,... 1月19日消息,目前最先进的量子计算机约有1000个量子比特,而美国哥伦比亚大学团队正试图将这一数字...
原创 青... 青海大通县17岁女孩的失联,牵动了所有人的心,已经上了高中的女孩马骕,最后的消失地点让大家心里仿佛覆...
重庆涪陵区:“三驾齐驱”构建金... 近年来,重庆涪陵区以提升企业获取金融服务满意度为核心,坚持政府引导、市场主导、机构协同,创新推动政策...
德国与乌克兰将联合生产长航程无... 据新华社基辅5月11日电(记者李东旭)正在乌克兰首都基辅访问的德国国防部长鲍里斯·皮斯托里乌斯11日...
特朗普:非常期待中国之行 当地时间5月11日,美国总统特朗普在其社交媒体“真实社交”发文表示,非常期待即将开启的中国之行,并称...