#include <iostream>
#include <mutex>
#include <condition_variable>
#include <thread>
#include <queue>
#include <chrono>
#include <vector>
using namespace std;
mutex mtx;/*两个线程间多个线程间进行共享变量的互斥*/
condition_variable cv;/*用于生产者线程和消费者线程相互通知,但是他没有锁的功能,只具有通知功能*/
queue<int> bufferQueue; /*生产者和消费者共享的buffer_queue,生产者方数据进去,消费者从里面拿数据*/
#define PRODUCTOR_BUFFER_NUM 10
int dataValue = 0; /*放在queue中的数据内容,依次递增*/
void Productor(){
while(1){/*线程一直不退出*/
std::this_thread::sleep_for(chrono::milliseconds(100));
std::unique_lock<std::mutex> lk(mtx);/*这个互斥量是用于互斥bufferQueue的push pop的*/
/*生产者线程一直阻塞等待通知,一旦被通知(说明bufferQueue不满了已经有被消费了),但是为了防止OS的假通知,这里
得到通知后,还需要在while中检查bufferQueue的是否满的,如果还是满的继续等待否则执行下面把数据添加到bufferQueue中操作
添加后然后在通知notify消费者来来消费*/
while(bufferQueue.size() >= PRODUCTOR_BUFFER_NUM){
cv.wait(lk);
}
/*如果bufferQueue里面可以添加数据,则Productor继续放数据*/
cout<<"Productor data:"<<dataValue<<" buffer size:"<<bufferQueue.size()<<endl;
bufferQueue.push(dataValue++);
/*把数据放在队列中后,通知consumer取数据*/
cv.notify_all();
}
}
void Consumer(){
while(1){/*线程一直不退出*/
std::this_thread::sleep_for(std::chrono::milliseconds(100));
std::unique_lock <std::mutex> lk(mtx);
/*当Consumer线程阻塞等到到被通知后(此时说明bufferQueue已经有数据了),但是为了方式被假唤醒,
所以这里需要在cv.wait后再次在while中检查下bufferQueue是否有数据,如果还是为空继续等待,否则
执行while下面代码逻辑消费bufferQueue*/
while(bufferQueue.empty()){
cv.wait(lk);
}
/*否则继续消费bufferQueque内容*/
cout<<"Cosume data:"<<bufferQueue.front()<<" buffer size:"<<bufferQueue.size()<<endl;
bufferQueue.pop();
cv.notify_all();
}
}
int main(){
vector<std::thread> threads;
threads.push_back(std::thread(&Productor));
threads.push_back(std::thread(&Productor));
threads.push_back(std::thread(&Productor));
threads.push_back(std::thread(&Consumer));
for(auto &t:threads){
t.join();
}
}