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 】
相关推荐:
Laravel如何生成URL和重定向?(路由助手函数)
EditPlus中的正则表达式 实战(1)
Laravel怎么创建自己的包(Package)_Laravel扩展包开发入门到发布
深圳网站制作培训,深圳哪些招聘网站比较好?
高防服务器租用指南:配置选择与快速部署攻略
laravel服务容器和依赖注入怎么理解_laravel服务容器与依赖注入解析
使用豆包 AI 辅助进行简单网页 HTML 结构设计
微信推文制作网站有哪些,怎么做微信推文,急?
Windows驱动无法加载错误解决方法_驱动签名验证失败处理步骤
如何挑选最适合建站的高性能VPS主机?
Laravel如何实现API资源集合?(Resource Collection教程)
详解CentOS6.5 安装 MySQL5.1.71的方法
为什么要用作用域操作符_php中访问类常量与静态属性的优势【解答】
Laravel Livewire是什么_使用Laravel Livewire构建动态前端界面
VIVO手机上del键无效OnKeyListener不响应的原因及解决方法
Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议
如何在沈阳梯子盘古建站优化SEO排名与功能模块?
Claude怎样写结构化提示词_Claude结构化提示词写法【教程】
制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?
Laravel数据库迁移怎么用_Laravel Migration管理数据库结构的正确姿势
Laravel如何使用Service Provider注册服务_Laravel服务提供者配置与加载
利用python获取某年中每个月的第一天和最后一天
香港服务器建站指南:免备案优势与SEO优化技巧全解析
如何在服务器上三步完成建站并提升流量?
Python文件操作最佳实践_稳定性说明【指导】
Laravel用户密码怎么加密_Laravel Hash门面使用教程
Laravel怎么多语言本地化设置_Laravel语言包翻译与Locale动态切换【手册】
HTML透明颜色代码怎么让图片透明_给img元素加透明色的技巧【方法】
如何在VPS电脑上快速搭建网站?
Laravel如何与Pusher实现实时通信?(WebSocket示例)
laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法
EditPlus中的正则表达式 实战(2)
5种Android数据存储方式汇总
Laravel Eloquent访问器与修改器是什么_Laravel Accessors & Mutators数据处理技巧
西安市网站制作公司,哪个相亲网站比较好?西安比较好的相亲网站?
php json中文编码为null的解决办法
网站制作大概要多少钱一个,做一个平台网站大概多少钱?
如何在IIS中新建站点并配置端口与物理路径?
laravel怎么使用数据库工厂(Factory)生成带有关联模型的数据_laravel Factory生成关联数据方法
Laravel的路由模型绑定怎么用_Laravel Route Model Binding简化控制器逻辑
如何用PHP快速搭建高效网站?分步指南
如何在 Go 中优雅地映射具有动态字段的 JSON 对象到结构体
,怎么在广州志愿者网站注册?
高端智能建站公司优选:品牌定制与SEO优化一站式服务
如何用花生壳三步快速搭建专属网站?
如何快速搭建高效服务器建站系统?
如何在IIS中新建站点并解决端口绑定冲突?
使用spring连接及操作mongodb3.0实例
武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?
浅析上传头像示例及其注意事项


