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


相关推荐: 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实例  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  浅析上传头像示例及其注意事项