Linux多线程编程怎么实现
发布时间 - 2023-05-19 00:00:00 点击率:次引言
条件变量是利用线程间共享的全局变量进行同步的一种机制,主要包括两个动作:一个线程等待条件变量的条件成立而挂起(此时不再占用cpu);另一个线程使条件成立(给出条件成立信号)。为了防止竞争,条件变量的使用总是和一个互斥锁结合在一起。
函数原型
1. 定义条件变量
#include/* 定义两个条件变量 */ pthread_cond_t cond_pro, cond_con;
2. 初始化和销毁条件变量
#includeint pthread_cond_init(pthread_cond_t *restrict cond, const pthread_condattr_t *restrict attr);int pthread_cond_destroy(pthread_cond_t *cond); /* 初始化条件变量 */ pthread_cond_init(&cond_pro, null); pthread_cond_init(&cond_con, null); /* 销毁条件变量 */ pthread_cond_destroy(&cond_pro); pthread_cond_destroy(&cond_pro);
3. 等待和激发条件
#includeint pthread_cond_wait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex); int pthread_cond_broadcast(pthread_cond_t *cond); int pthread_cond_signal(pthread_cond_t *cond); /* 等待条件 */ /* 注意:pthread_cond_wait为阻塞函数。解开锁,再等待。等条件满足时,需要抢到锁,才可以被唤醒*/ pthread_cond_wait(&cond_pro,&mutex); /* 激发条件 */ /* 所有因为不满足条件的线程都会阻塞在条件变量cond_pro中的一个队列中 */ /* 以广播方式,通知所有被阻塞的所有线程 */ pthread_cond_broadcast(&cond_pro); /* 以signal方式,只通知排在最前面的线程 */ pthread_cond_signal(&cond_pro);
代码
/************************************************************************* > file name: my_con.c > author: krischou > mail:zhoujx0219@163.com > created time: tue 26 aug 2014 10:24:29 am cst ************************************************************************/ #include#include #include #include #include #define cell 10 #define flore 0 int i = 0; /* 所有线程共享的全局变量,此处假定至多递增至10,最小减到0 */ pthread_mutex_t mutex; /* 定义互斥锁 */ pthread_cond_t cond_pro, cond_con; /* 定义两个条件变量 */ /* 生产者线程 */ void* pro_handler(void *arg) { pthread_detach(pthread_self()); /* 由系统回收线程资源,而非主线程回收资源 ,此类情况主线程是个服务器,永久不会退出 */ while(1) { pthread_mutex_lock(&mutex); while(i >= cell) { pthread_cond_wait(&cond_pro,&mutex); /* continue是轮询,此处是阻塞 */ /* 把锁放开再等 ,第一个参数是结构体指针,其中有成员存放被阻塞的函数 */ /*不占cpu*/ /* 不满足条件时才会等 ,需要别人告诉它,才能唤醒它*//* 当它返回时,锁也要回来了*/ } i++; if(i == 1) { /* 由空到不空,唤醒消费者 */ pthread_cond_signal(&cond_con); /*不会立马signal被阻塞的消费者线程,因为其还要等锁抢回来*/ } printf("add i: %d \n", i); pthread_mutex_unlock(&mutex); sleep(rand() % 5 + 1); } } /* 消费者线程 */ void* con_handler(void *arg) { pthread_detach(pthread_self()); while(1) { pthread_mutex_lock(&mutex); while(i <= flore) { pthread_cond_wait(&cond_cno,&mutex); } i--; if(i == 9) /* 由满到不满,要告诉生产者,以便将其唤醒 *//*此处,直接signal也可以,我们是为了更加精确*/ { pthread_cond_signal(&cond_pro); } printf("con i: %d \n", i); pthread_mutex_unlock(&mutex); sleep(rand() % 5 + 1); } } int main(int argc, char *argv[]) // exe +num -num { srand(getpid()); int con_cnt, pro_cnt; pro_cnt = atoi(argv[1]); con_cnt = atoi(argv[2]); pthread_mutex_init(&mutex,null); pthread_cond_init(&cond_pro,null); pthread_cond_init(&cond_con,null); pthread_t *arr = (pthread_t*)calloc(con_cnt + pro_cnt , sizeof(pthread_t)); int index = 0; while(pro_cnt > 0) { pthread_create(arr + index, null, pro_handler, null); index++; pro_cnt--; } while(con_cnt > 0) { pthread_create(arr + index, null, con_handler, null); index++; con_cnt--; } while(1); pthread_mutex_destroy(&mutex); pthread_cond_destroy(&cond_pro); pthread_cond_destroy(&cond_con); return 0; }
注意
无论是在生产者线程,还是在消费者线程中。标记黄色部分的判断条件必须用while。以生产者线程举例,当i>=cell时,也就是i满时,此时执行pthread_cond_wait(&cond_cno,&mutex); 该生产者线程被挂起。必须等到消费者线程pthread_cond_signal(&cond_pro); 将其唤醒。但是消费者将其signal还不够,被挂其的生产者线程必须重新拿到锁,才可以被激活。但是,由于在消费者signal的同时,生产者并不能立即抢到锁,所以此时可能i值又改变变为大于等于10了。因此必须用while。不然可能导致i>10。
# linux
# while
# 全局变量
# signal
# 线程
# 多线程
# 将其
# 是在
# 才可以
# 抢到
# 再等
# 不满足
# 挂起
# 是个
# 来了
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
js实现获取鼠标当前的位置
猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?
Laravel如何使用withoutEvents方法临时禁用模型事件
东莞市网站制作公司有哪些,东莞找工作用什么网站好?
php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】
Win11怎么关闭资讯和兴趣_Windows11任务栏设置隐藏小组件
Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录
如何在服务器上配置二级域名建站?
手机网站制作与建设方案,手机网站如何建设?
如何生成腾讯云建站专用兑换码?
ChatGPT回答中断怎么办 引导AI继续输出完整内容的方法
javascript日期怎么处理_如何格式化输出
ChatGPT 4.0官网入口地址 ChatGPT在线体验官网
教你用AI将一段旋律扩展成一首完整的曲子
Laravel如何使用.env文件管理环境变量?(最佳实践)
HTML5空格在Angular项目里怎么处理_Angular中空格的渲染问题【详解】
微信小程序 HTTPS报错整理常见问题及解决方案
uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址
,网页ppt怎么弄成自己的ppt?
简历在线制作网站免费版,如何创建个人简历?
Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)
百度浏览器网页无法复制文字怎么办 百度浏览器复制修复
音响网站制作视频教程,隆霸音响官方网站?
Python制作简易注册登录系统
如何挑选高效建站主机与优质域名?
如何在 Go 中优雅地映射具有动态字段的 JSON 对象到结构体
Laravel如何构建RESTful API_Laravel标准化API接口开发指南
Laravel辅助函数有哪些_Laravel Helpers常用助手函数大全
如何在VPS电脑上快速搭建网站?
Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法
重庆市网站制作公司,重庆招聘网站哪个好?
Laravel如何实现API版本控制_Laravel API版本化路由设计策略
如何实现javascript表单验证_正则表达式有哪些实用技巧
Laravel如何使用模型观察者?(Observer代码示例)
JavaScript如何实现倒计时_时间函数如何精确控制
Laravel如何实现本地化和多语言支持_Laravel多语言配置与翻译文件管理
Laravel怎么进行浏览器测试_Laravel Dusk自动化浏览器测试入门
Laravel如何使用Laravel Vite编译前端_Laravel10以上版本前端静态资源管理【教程】
Laravel怎么实现软删除SoftDeletes_Laravel模型回收站功能与数据恢复【步骤】
Linux安全能力提升路径_长期防护思维说明【指导】
Laravel storage目录权限问题_Laravel文件写入权限设置
Laravel怎么实现验证码(Captcha)功能
详解jQuery中基本的动画方法
Laravel如何创建自定义中间件?(Middleware代码示例)
安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出
laravel怎么为应用开启和关闭维护模式_laravel应用维护模式开启与关闭方法
网站建设整体流程解析,建站其实很容易!
音乐网站服务器如何优化API响应速度?
北京网站制作公司哪家好一点,北京租房网站有哪些?
QQ浏览器网页版登录入口 个人中心在线进入


