之前写过一篇关于多线程的https://blog.csdn.net/qq_21049875/article/details/79589126。
  为了复习一下C++多线程的使用,以及程序的编写,于是写了一个生产者与消费者。
  在编写的过程中了解了条件变量(std::condition_variable)的使用,可以让线程挂起和叫醒线程,以及之前编写一些示例代码的时候一直用的是std::locak_guard,而不是std::unique_lock,两者都是RAII(获取资源立即初始化 ),而后者比较灵活,提供了lock(),unlock(),try_lock(),接口。
  std::unique_lock的源代码中大概是封装了mutex的一些接口,下面从其他博客复制了mutex的接口信息。)
  lock(),调用线程将锁住该互斥量。线程调用该函数会发生下面 3 种情况:
    (1). 如果该互斥量当前没有被锁住,则调用线程将该互斥量锁住,直到调用 unlock之前,该线程一直拥有该锁。
    (2). 如果当前互斥量被其他线程锁住,则当前的调用线程被阻塞住。
    (3). 如果当前互斥量被当前调用线程锁住,则会产生死锁(deadlock)。
  unlock(), 解锁,释放对互斥量的所有权。
  try_lock(),尝试锁住互斥量,如果互斥量被其他线程占有,则当前线程也不会被阻塞。线程调用该函数也会出现下面 3 种情况:
    (1). 如果当前互斥量没有被其他线程占有,则该线程锁住互斥量,直到该线程调用 unlock 释放互斥量。
    (2). 如果当前互斥量被其他线程锁住,则当前调用线程返回 false,而并不会被阻塞掉。
    (3). 如果当前互斥量被当前调用线程锁住,则会产生死锁(deadlock)。

  下面代码编写中碰到的问题,在productItem与consumeItem的时候,都有对互斥量mtx上锁的过程,在productItem的代码里有一个Sleep,是为了让消费者能够消费到商品,要不然就由于计算机速度太快,直接生产满了,就碰到while(item…)直接出发is_full造成后面运行后生产者进入死锁状态。
  当然,你可以编写一个对is_full这个条件变量相应的函数,比如item满了就扔掉一个,去解锁is_full然后唤醒product线程,这样就不需要Sleep函数啦~
  还有,可以用std::this_thread::yield()来实现把执行的线程时间片让给其他线程使用,就比如item满了之后让其他线程运行,或者是std::this_thread::sleep_for() 让线程睡一下,更或者直接sleep……hah….

代码:

#include<iostream>
#include<vector>
#include<thread>
#include<queue>
#include<mutex>
#include<Windows.h>

class demo {
private:
condition_variable is_full;
condition_variable is_empty;
mutex mtx;
vector<std::thread> tpool;
private:
queue<int> item_buff;
const int buffsize;
static int item_id;
int c_no, p_no;
private:
void producItem() {
while (true) {
std::unique_lock<mutex> lck(mtx);
while (item_buff.size() == buffsize) {
is_full.wait(lck);
}
item_id = item_id++ %buffsize;
item_buff.push(item_id);
cout << "product item id: " << item_id << endl;
lck.unlock();
is_empty.notify_all();
Sleep(10);
}
}
void consumeItem() {
while (true) {
std::unique_lock<mutex> lck(mtx);
while (item_buff.empty()) {
cout << "wait product" << endl;
is_full.notify_one();
is_empty.wait(lck);
}
cout << "customer id : " << this_thread::get_id() << endl;
cout << "consume Item id: " << item_buff.front() << endl;
item_buff.pop();
lck.unlock();
}
}
public:
demo(int size = 10, int pno = 1, int cno = 5) :buffsize(size), p_no(pno), c_no(cno) {};
void run() {
productTask();
customerTask();
for (auto &i : tpool) {
i.join();
}
}
void productTask() {
for (int i = 0; i < p_no; i++) {
tpool.emplace_back(thread(&demo::producItem, this));
}
}
void customerTask() {
for (int i = 0; i < c_no; i++) {
tpool.emplace_back(thread(&demo::consumeItem, this));
}
}
};

int demo::item_id = 0;

int main()
{
demo t;
t.run();
return 0;
}

C++多线程 生产者 消费者示例的更多相关文章

  1. java+反射+多线程+生产者消费者模式+读取xml(SAX)入数据库mysql-【费元星Q9715234】

    java+反射+多线程+生产者消费者模式+读取xml(SAX)入数据库mysql-[费元星Q9715234] 说明如下,不懂的问题直接我[费元星Q9715234] 1.反射的意义在于不将xml tag ...

  2. Java阻塞队列(BlockingQueue)实现 生产者/消费者 示例

    Java阻塞队列(BlockingQueue)实现 生产者/消费者 示例 本文由 TonySpark 翻译自 Javarevisited.转载请参见文章末尾的要求. Java.util.concurr ...

  3. python自动化--语言基础线程、生产者消费者示例

    进程与线程的区别:进程不共享空间,线程共享地址空间 线程共享空间优缺点:优点:多线程给用户的体验好些,打开时占用的内存比进程少缺点:共享地址空间会相互干扰,甚至有影响 import threading ...

  4. Python多线程-生产者消费者模型

    用多线程和队列来实现生产者消费者模型 # -*- coding:utf-8 -*- __author__ = "MuT6 Sch01aR" import threading imp ...

  5. Java实现多线程生产者消费者模式的两种方法

    生产者消费者模式:生产者和消费者在同一时间段内共用同一存储空间,生产者向空间里生产数据,而消费者取走数据.生产者生产一个,消费者消费一个,不断循环. 第一种实现方法,用BlockingQueue阻塞队 ...

  6. 多线程-生产者消费者(synchronized同步)

    正解博客:https://blog.csdn.net/u011863767/article/details/59731447 永远在循环(loop)里调用 wait 和 notify,不是在 If 语 ...

  7. java多线程 生产者消费者模式

    package de.bvb; /** * 生产者消费者模式 * 通过 wait() 和 notify() 通信方法实现 * */ public class Test1 { public static ...

  8. [多线程] 生产者消费者模型的BOOST实现

    说明 如果 使用过程中有BUG 一定要告诉我:在下面留言或者给我邮件(sawpara at 126 dot com) 使用boost::thread库来实现生产者消费者模型中的缓冲区! 仓库内最多可以 ...

  9. Java实现多线程生产者消费者模型及优化方案

    生产者-消费者模型是进程间通信的重要内容之一.其原理十分简单,但自己用语言实现往往会出现很多的问题,下面我们用一系列代码来展现在编码中容易出现的问题以及最优解决方案. /* 单生产者.单消费者生产烤鸭 ...

随机推荐

  1. yum安装nagois

    多少年前就装过了,今天再来用yum装一次,都忘干净了~~ 主监控机:CentOS 6.5 192.168.0.105被监控机:CentOS 6.5 192.168.0.107 主监控机设置:1.安装e ...

  2. linux nginx安装以及配置

    一.Nginx简介 Nginx (“engine x”) 是一个高性能的HTTP和反向代理服务器,也是一个IMAP/POP3/SMTP服务器.Nginx是由Igor Sysoev为俄罗斯访问量第二的R ...

  3. Excel2013复制内容粘贴到刷选的数据表中

    如何将复制内容粘贴到筛选后表格中: 1.需要一个表格:我要把这个表格里姓名列不包含A的项改为另外一个表中的数据列: 2.在姓名右侧插入两列,列A.列B: 3.在列A下第一格输入1,向下序列填充: 4. ...

  4. Flume学习总结

    Flume学习总结 flume是一个用来采集数据的软件,它可以从数据源采集数据到一个集中存放的地方. 最常用flume的数据采集场景是对日志的采集,不过,lume也可以用来采集其他的各种各样的数据,因 ...

  5. IIS上架设https网站证书处理备忘

    1. 免费SSL证书申请 https://www.startssl.com 教程:http://hxs.fd.fj.cn/?action=show&id=13 2. 证书转换 申请到的证书有两 ...

  6. 手游热更新方案xLua开源:Unity3D下Lua编程解决方案

    C#下Lua编程支持 xLua为Unity. .Net. Mono等C#环境增加Lua脚本编程的能力,借助xLua,这些Lua代码可以方便的和C#相互调用. xLua的突破 xLua在功能.性能.易用 ...

  7. bootstrop-datatime参数配置

    <html> <head> <meta http-equiv="Content-Type" content="text/html; char ...

  8. IDEA07 快捷键统计插件之Key Promoter X

    1 官方介绍 该插件会统计你鼠标操作的次数,然后在一个窗口中展示出来 https://plugins.jetbrains.com/plugin/9792-key-promoter-x 2 下载.安装. ...

  9. ultraiso制作ubuntu u盘启动

    http://blog.csdn.net/yaoyut/article/details/78003061

  10. struts2下velocity做视图如何访问request,session等内置对象,如:原来webwork的$req

    struts2下velocity做视图如何访问request,session等内置对象(转) velocity 内置对象 struts2 requestStruts2环境下用velocity做视图时访 ...