Linux多线程编程怎么实现

发布时间 - 2023-05-19 00:00:00    点击率:

引言
条件变量是利用线程间共享的全局变量进行同步的一种机制,主要包括两个动作:一个线程等待条件变量的条件成立而挂起(此时不再占用cpu);另一个线程使条件成立(给出条件成立信号)。为了防止竞争,条件变量的使用总是和一个互斥锁结合在一起。

函数原型
1. 定义条件变量

#include 

/* 定义两个条件变量 */
pthread_cond_t cond_pro, cond_con;

2. 初始化和销毁条件变量

#include 

int 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. 等待和激发条件

#include 

int 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浏览器网页版登录入口 个人中心在线进入