当前位置: 首页 > news >正文

linux下实现生产者和消费者 pv操作

线程同步与线程安全

  • 生产者和消费者
    • 特点
    • 图示理解
    • 编程实现
    • 测试结果

生产者和消费者

特点

1.解耦:因为多了一个缓冲区,所以生产者和消费者并不直接相互调用,这样生产者和消费者的代码发生变化,都不会对对方产生影响。这样其实就是把生产者和消费者之间的强耦合解开,变成了生产者和缓冲区,消费者和缓冲区之间的弱耦合
取快递,快递放到菜鸟驿站,
⒉.支持并发:如果消费者直接从生产者拿数据,则消费者需要等待生产者生产数据,同样生产者需要等待消费者消费数据。而有了生产者/消费者模型,生产者和消费者可以是两个独立的并发主体。生产者把制造出来的数据添加到缓冲区,就可以再去生产下一个数据了。而消费者也是一样的,从缓冲区中读取数据,不需要等待生产者。这样,生产者和消费者就可以并发的执行。
3.支持忙闲不均:如果消费者直接从生产者这里拿数据,而生产者生产数据很慢,消费者消费数据很快,或者生产者生产数据很多,消费者消费数据很慢。都会造成占用CPU的时间片白白浪费。生产者/消费者模型中,生产者只需要将生产的数据添加到缓冲区,缓冲区满了就不生产了。消费者从缓冲区中读取数据,缓冲区空了就不消费了,使得生产者/消费者的处理能力达到一个动态的平衡。

图示理解

在这里插入图片描述

编程实现

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>
#include <semaphore.h>
#include <time.h>#define BUFF_MAX 30
#define SC_NUM 3
#define XF_NUM 2sem_t sc_sem;
sem_t xf_sem;
pthread_mutex_t mutex;int in = 0;
int out = 0;int Arr_Buff[BUFF_MAX];
//生产者线程
void *sc_fun(void *arg)
{int index = (int)arg;for (int i = 0; i < 20; i++){sem_wait(&sc_sem);pthread_mutex_lock(&mutex);Arr_Buff[in] = rand() % 100;printf("sc %d in %d write data %d\n", index, in, Arr_Buff[in]);in = (in + 1) % BUFF_MAX;pthread_mutex_unlock(&mutex);sem_post(&xf_sem);int n = rand() % 5;sleep(n);}
}
//消费者线程
void *xf_fun(void *arg)
{int index = (int)arg;for (int i = 0; i < 30; i++){sem_wait(&xf_sem);pthread_mutex_lock(&mutex);printf("-------xf %d in %d read data:%d\n", index, out, Arr_Buff[out]);out = (out + 1) % BUFF_MAX;pthread_mutex_unlock(&mutex);sem_post(&sc_sem);int n = rand() % 5;sleep(n);}
}
int main()
{sem_init(&sc_sem, 0, BUFF_MAX);sem_init(&xf_sem, 0, 0);pthread_mutex_init(&mutex, NULL);srand((int)time(NULL));pthread_t sc_id[SC_NUM];for (int i = 0; i < SC_NUM; i++){pthread_create(&sc_id[i], NULL, sc_fun, (void *)i);}pthread_t xf_id[XF_NUM];for (int i = 0; i < XF_NUM; i++){pthread_create(&xf_id[i], NULL, xf_fun, (void *)i);}for (int i = 0; i < SC_NUM; i++){pthread_join(sc_id[i], NULL);}for (int i = 0; i < XF_NUM; i++){pthread_join(xf_id[i], NULL);}pthread_mutex_destroy(&mutex);sem_destroy(&sc_sem);sem_destroy(&xf_sem);exit(0);
}

测试结果

在这里插入图片描述

http://www.lryc.cn/news/119045.html

相关文章:

  • 十六、遥感影像识别
  • 源码角度分析@configuration和@component不同
  • 实现分布式事务:Java与MySQL的XA事务协调
  • 如何通过CRM系统进行成功的客户生命周期管理?
  • Leetcode 977. 有序数组的平方
  • vue3中使用toValue
  • 阿里云国际版CDN使用教程!
  • 【docker】Dockerfile构建镜像常用指令:
  • Java课题笔记~ 会话跟踪
  • HbuildX生成安卓签名证书
  • 在Ruoyi中采用Ajax动态生成Echarts图表实践
  • 资讯速递 | ArkUI-X 预览版已正式开源!
  • HTML <select> 标签
  • Flutter:文件上传与下载(下载后预览)
  • [前端系列第6弹]Ajax简明教程:轻松实现Web页面的异步交互
  • ssh-keygen 做好免密登录后不生效
  • 【Java可执行命令】(十八)可视化监控和管理工具 jconsole:获取 JVM的内存使用情况、线程活动、GC 行为等重要指标的可视化工具 ~
  • leetcode做题笔记66
  • 【docker】设置 docker 国内镜像报错,解决方案
  • mac安装nvm管理工具遇到的问题和解决方法
  • DocX 生成Word
  • 数据库新闻速递 -- POSTGRESQL 正在蚕食数据库市场 (翻译)
  • PAT 1085 Perfect Sequence
  • 软件测试面试夺命连环十七问,你答得上来么?这都不会建议多学!
  • 【学习FreeRTOS】第5章——FreeRTOS任务挂起与恢复
  • gitblit-使用
  • 整数中1出现的次数(从1到n整数中1出现的次数)
  • Vue2:路由
  • 【Docker】Docker的应用场景,Docker 的优点,Ubuntu Docker 安装,使用 Shell 脚本进行安装
  • CentOS7 启动谷歌浏览器 java+Selenium+chrome+chromedriver