linux条件变量知识点总结

news/2024/11/29 12:32:39/

与条件变量相关API

    条件变量是线程另一可用的同步机制。条件变量给多个线程提供了一个会合的场所。条件变量与互斥量一起使用时,允许线程以无竞争的方式等待特定的条件发生。

  条件本身是由互斥量保护的。线程在改变条件状态前必须首先锁住互斥量,其他线程在获得互斥量之前不会察觉到这种改变,因为必须锁定互斥量以后才能计算条件。

  条件变量使用之前必须首先初始化,pthread_cond_t数据类型代表的条件变量可以用两种方式进行初始化,可以把常量PTHREAD_COND_INITIALIZER赋给静态分配的条件变量,但是如果条件变量是动态分配的,可以使用pthread_cond_destroy函数对条件变量进行去除初始化(deinitialize)。

1. 创建及销毁条件变量

#include <pthread.h>
int pthread_cond_init(pthread_cond_t *restrict cond, const pthread_condattr_t *restrict attr);
int pthread_cond_destroy(pthread_cond_t *cond);
// 返回:若成功返回0,否则返回错误编号

  除非需要创建一个非默认属性的条件变量,否则pthread_cont_init函数的attr参数可以设置为NULL。

2. 等待

#include <pthread.h>
int pthread_cond_wait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex);
int pthread_cond_timedwait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex, cond struct timespec *restrict timeout);
// 返回:若成功返回0,否则返回错误编号

  pthread_cond_wait等待条件变为真。如果在给定的时间内条件不能满足,那么会生成一个代表一个出错码的返回变量。传递给pthread_cond_wait的互斥量对条件进行保护,调用者把锁住的互斥量传给函数。函数把调用线程放到等待条件的线程列表上,然后对互斥量解锁,这两个操作都是原子操作。这样就关闭了条件检查和线程进入休眠状态等待条件改变这两个操作之间的时间通道,这样线程就不会错过条件的任何变化。pthread_cond_wait返回时,互斥量再次被锁住。

  pthread_cond_timedwait函数的工作方式与pthread_cond_wait函数类似,只是多了一个timeout。timeout指定了等待的时间,它是通过timespec结构指定。

3. 触发

#include <pthread.h>
int pthread_cond_signal(pthread_cond_t *cond);
int pthread_cond_broadcast(pthread_cond_t *cond);
// 返回:若成功返回0,否则返回错误编号

  这两个函数可以用于通知线程条件已经满足。pthread_cond_signal函数将唤醒等待该条件的某个线程,而pthread_cond_broadcast函数将唤醒等待该条件的所有进程。

  注意一定要在改变条件状态以后再给线程发信号。

Linux线程开发中,通常对于互斥量、条件变量等都会提供两种初始化方式,分别是动态初始化和静态初始化。

一般来说变量的静态初始是预编译时对变量进行初始化,而动态初始,是在程序运行时,根据需要,动态分配空间并赋值

如:
pthread cond t cond;
//动态初始化: pthread cond init(&cond,NULL);
/静态初始化: pthread cond t = PTHREAD COND INITIALIZER;

pthread mutex_t;
//动态初始化: pthread mutex init(&mutex,NULL);
//静态初始化: pthread mutex tT= PTHREAD MUTEX INITIALIZER;

示例

(1)线程条件控制实现线程同步(动态初始化)

#include <stdio.h>
#include <pthread.h>
#include <stdlib.h>//int pthread_create(pthread_t *restrict tidp, const pthread_attr_t *restrict attr, void *(*start_rtn)(void *), void *restrict arg);
int g_data = 0;pthread_mutex_t mutex;
pthread_cond_t cond;void *func1(void *arg)
{printf("t1:%ld thread is create\n",(unsigned long)pthread_self());printf("t1:param is %d\n",*((int *)arg));static int cnt = 0;while(1){pthread_cond_wait(&cond,&mutex);printf("t1 run================================\n");printf("t1: %d\n",g_data);    g_data = 0;sleep(1);if(cnt++ == 10){exit(1);}}}void *func2(void *arg)
{printf("t2:%ld thread is create\n",(unsigned long)pthread_self());printf("t2:param is %d\n",*((int *)arg));while(1){printf("t2: %d\n",g_data);pthread_mutex_lock(&mutex);g_data++;if(g_data == 3){pthread_cond_signal(&cond);}pthread_mutex_unlock(&mutex);    sleep(1);}
}int main()
{int ret;int param = 100;pthread_t t1;pthread_t t2;//动态初始化pthread_mutex_init(&mutex,NULL);pthread_cond_init(&cond,NULL);ret = pthread_create(&t1, NULL, func1,(void *)&param);if(ret == 0){
//        printf("main:create t1 success\n");}ret = pthread_create(&t2, NULL, func2,(void *)&param);if(ret == 0){
//        printf("main:create t2 success\n");}//    printf("main:%ld\n",(unsigned long)pthread_self());pthread_join(t1,NULL);pthread_join(t2,NULL);pthread_mutex_destroy(&mutex);pthread_cond_destroy(&cond);return 0;
}

编译结果:

(2)线程条件控制实现线程同步(静态初始化)

#include <stdio.h>
#include <pthread.h>
#include <stdlib.h>//int pthread_create(pthread_t *restrict tidp, const pthread_attr_t *restrict attr, void *(*start_rtn)(void *), void *restrict arg);
int g_data = 0;//静态初始化
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;void *func1(void *arg)
{printf("t1:%ld thread is create\n",(unsigned long)pthread_self());printf("t1:param is %d\n",*((int *)arg));static int cnt = 0;while(1){pthread_cond_wait(&cond,&mutex);printf("t1 run================================\n");printf("t1: %d\n",g_data);    g_data = 0;sleep(1);if(cnt++ == 10){exit(1);}}}void *func2(void *arg)
{printf("t2:%ld thread is create\n",(unsigned long)pthread_self());printf("t2:param is %d\n",*((int *)arg));while(1){printf("t2: %d\n",g_data);pthread_mutex_lock(&mutex);g_data++;if(g_data == 3){pthread_cond_signal(&cond);}pthread_mutex_unlock(&mutex);    sleep(1);}
}int main()
{int ret;int param = 100;pthread_t t1;pthread_t t2;//动态初始化
//    pthread_mutex_init(&mutex,NULL);
//    pthread_cond_init(&cond,NULL);ret = pthread_create(&t1, NULL, func1,(void *)&param);if(ret == 0){
//        printf("main:create t1 success\n");}ret = pthread_create(&t2, NULL, func2,(void *)&param);if(ret == 0){
//        printf("main:create t2 success\n");}//    printf("main:%ld\n",(unsigned long)pthread_self());pthread_join(t1,NULL);pthread_join(t2,NULL);pthread_mutex_destroy(&mutex);pthread_cond_destroy(&cond);return 0;
}


编译结果:与上面一样


http://www.ppmy.cn/news/109645.html

相关文章

高速缓存(cache)的原理: 了解计算机架构与性能优化

计基之存储器层次结构 Author&#xff1a; Once Day Date&#xff1a; 2023年5月9日 长路漫漫&#xff0c;而今才刚刚启程&#xff01; 本内容收集整理于《深入理解计算机系统》一书。 参看文档: 捋一捋Cache - 知乎 (zhihu.com)iCache和dCache一致性 - 知乎 (zhihu.com)C…

使用SolidJS+SpringBoot写一个简单的个人博客

首发于Enaium的个人博客 前言 前端我们使用了SolidJS来进行开发,它是一个新出的前端框架,它的特点是使用React Hooks的方式来进行开发,并且它的API和React的API非常相似,所以如果你会React的话,那么你就会SolidJS. 后端我们使用了SpringBoot来进行开发,数据库我们使用了MySQL…

IMX6ULL裸机篇之I2C相关寄存器

一. I2C实验 I2C时钟选择与传输速率 1. IMX6ULL的 I2C频率标准模式 100kbit/S&#xff0c;快速模式为 400Kbit/S 2. 时钟源选择 perclk_clk_rootipg_clk_root66MHz&#xff08;由之前的时钟实验章节可以知道是 66MHz&#xff09;。 二. I2C 寄存器配置 I2Cx_IFDR寄存器&…

考研C语言第七章

7.2函数的声明定义&#xff0c;函数的分类和调用 1、函数声明要在main函数上面&#xff0c;这样main函数才可以进行调用 第一节课主要就学了函数的调用&#xff0c;项目在开发的时候分工用的比较多 主要结构&#xff1a; func.c里面进行函数的定义 不是很重要&#xff0c;进…

VUE3浅析---响应式

VUE3中setup语法糖解决响应式的方案&#xff0c;所有的只要被ref或者reactive包裹的变量&#xff0c;都会转变成响应式。而在VUE2中&#xff0c;要想做成响应式&#xff0c;必须将变量定义在data函数中。 文章目录 1、ref&#xff1a;将一个属性或者对象定义成ref对象&#xff…

C语言杂七杂八

fget函数的用法 fgets() 函数用来从指定的文件中读取一个字符串&#xff0c;并保存到字符数组中。 str 为字符数组&#xff0c;n 为要读取的字符数目&#xff0c;fp 为文件指针。 char *fgets(char *str, int n, FILE *stream) 返回值&#xff1a;读取成功时返回字符数组首地…

Ceres简介及示例(10)On Derivatives(Automatic Derivatives)

现在我们将讨论自动微分算法。它是一种可以快速计算精确导数的算法&#xff0c;同时用户只要做与数值微分法类似的工作。下面的代码片段实现了对Rat43的CostFunction。 struct Rat43CostFunctor {Rat43CostFunctor(const double x, const double y) : x_(x), y_(y) {}template…

象棋游戏(文案)

绿王和红王的战斗规则绿王和红王的战斗规则刘邦和项羽&#xff0c;楚河汉界&#xff0c;红王和绿王的象棋游戏世界。  方案规则&#xff1b;  1、楚河汉界中有一口水井&#xff0c;这是战场的最后归宿&#xff0c;气血值是由这一口水井进行分配的。&#xff08;地图造型水井…