QT QThread +信号量 实现生成者和消费者

30 篇文章 0 订阅

//本文详细描述QT 中QThread +信号量 实现生成者和消费者。

//调试通过。

//这个例子演示了怎样使用QSemaphore 信号量来保护对生成者线程和消费者线程共享的环形缓冲

//区的访问。
//生成者向缓冲区中写入数据,直到达到缓冲区的终点,这时它会从起点重新开始,覆盖已经存在

//的数据
//消费者线程读取它产生的数据,并将其输出。
//这个例子中包含Producer和Consumer类,它们都继承自QTherad
//环形缓冲区用来在这两个缓冲区通讯,保护缓冲区的信号量被设置为全局变量

//---------------------------------------------------------------------

//1.main 文件

#include <QCoreApplication>
#include <QSemaphore>
#include "producer.h"
#include "consumer.h"
#include <QDebug>
int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);
    qDebug()<<"start";
    Producer producer;
    Consumer consumer;
    /* 启动生产者和消费者线程 */
    producer.start();
    consumer.start();
    /* 等待生产者和消费者各自执行完毕后自动退出 */
    producer.wait();
    consumer.wait();

    qDebug()<<"end";
    return a.exec();
}

//---------------------------------------------------------------------

//---------------------------------------------------------------------

//2.全局变量

//2.1全局变量头文件

#include <QSemaphore>
#ifndef GLOBAL_DEF_H
#define GLOBAL_DEF_H
extern const int DataSize;//生产者将要产生数据的数量
extern const int BufferSize;//环形缓冲区的大小
extern char buffer [];//缓冲区
extern QSemaphore freeBytes;//控制缓冲区的空闲区域(生成者还没添加数据或者消费者已经进

//行读取的区域)
extern QSemaphore usedBytes;//控制已经使用了的缓冲区区域(生成者已经添加数据,但是消费

//者还没有读取的区域)
#endif // GLOBAL_DEF_H

//---------------------------------------------------------------------

//2.2 全局变量cpp文件

#include "global_def.h"
const int DataSize=10;//生产者将要产生数据的数量
const int BufferSize=5;//环形缓冲区的大小
char buffer [BufferSize];//缓冲区
QSemaphore freeBytes(BufferSize);//控制缓冲区的空闲区域(生成者还没添加数据或者消费者已经进行读取的区域)
QSemaphore usedBytes(0);//控制已经使用了的缓冲区区域(生成者已经添加数据,但是消费者还没有读取的区域)
//这两个信号保证了生成者永远不会在消费者前多于BufferSize个字节,而消费者永远也不会读取生成者还没生成的数据

//---------------------------------------------------------------------

//---------------------------------------------------------------------
//3.生成者类

//3.1 生成者类头文件

#ifndef PRODUCER_H
#define PRODUCER_H
#include <QThread>
#include "global_def.h"
#include <QRandomGenerator>
#include <QDebug>
class Producer : public QThread
{
public:
     Producer();
     void run() ;//注意大小写,不是Run()函数
};
#endif // PRODUCER_H

//---------------------------------------------------------------------
//3.2 生成者类cpp文件

#include "producer.h"
Producer::Producer()
{}
void Producer::run()
{
    for(int i=0;i<DataSize;i++)
    {
        freeBytes.acquire();
        int randomInt = QRandomGenerator::global()->bounded(100);
        buffer[i % BufferSize] ="ACGT"[(int) randomInt % 4 ];
        qDebug()<<QString("producer:%1").arg(buffer[i % BufferSize]);

        usedBytes.release();
    }
}

//---------------------------------------------------------------------

//---------------------------------------------------------------------
//4. 消费者类 

//4.1 消费者类 头文件

#ifndef CONSUMER_H
#define CONSUMER_H

#include <QThread>
#include <QDebug>
#include "global_def.h"
class Consumer : public QThread
{
public:
    Consumer();
     void run() ;//注意大小写,不是Run()函数
};
#endif // CONSUMER_H

//---------------------------------------------------------------------
//4.2 消费者类 cpp文件

#include "consumer.h"

Consumer::Consumer()

{}
void Consumer::run()
{
    for(int i=0;i<DataSize;i++)
    {
        usedBytes.acquire();
        qDebug()<<QString("consumer:%1").arg(buffer[i % BufferSize]);
        freeBytes.release();
    }
}

//---------------------------------------------------------------------

  • 11
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Qt提供了QSemaphore类来实现信号量功能,可以用于实现一个生产者多个消费者的场景。 首先,我们需要创建一个QSemaphore对象,作为生产者和消费者之间共享的信号量。在这个例子中,我们假设生产者线程负责生产产品,消费者线程负责消费产品。 生产者的逻辑如下: 1. 首先,获取信号量的锁定,如果信号量的计数器为0,则阻塞等待信号量的释放; 2. 当信号量的计数器不为0时,生产者线程开始生产产品,并进行相应的操作; 3. 完成产品的生产后,释放信号量的锁定,并增加信号量的计数器。 消费者的逻辑如下: 1. 首先,获取信号量的锁定,如果信号量的计数器为0,则阻塞等待信号量的释放; 2. 当信号量的计数器不为0时,消费者线程开始消费产品,并进行相应的操作; 3. 完成产品的消费后,释放信号量的锁定,并增加信号量的计数器。 下面是一个简化的示例代码: ```cpp QSemaphore semaphore; // 创建信号量对象 QVector<QString> products; // 存放产品的容器 // 生产者线程函数 void producer() { while (true) { semaphore.acquire(); // 获取信号量的锁定 // 生产产品的操作 QString product = generateProduct(); products.append(product); semaphore.release(); // 释放信号量的锁定 QThread::sleep(1); // 等待一段时间 } } // 消费者线程函数 void consumer() { while (true) { semaphore.acquire(); // 获取信号量的锁定 // 消费产品的操作 if (!products.isEmpty()) { QString product = products.takeFirst(); consumeProduct(product); } semaphore.release(); // 释放信号量的锁定 QThread::sleep(1); // 等待一段时间 } } int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); // 创建生产者和消费者线程 QThread producerThread; QThread consumerThread1; QThread consumerThread2; QThread consumerThread3; // 信号量关联线程 semaphore.setInitialValue(1); semaphore.moveToThread(&producerThread); semaphore.moveToThread(&consumerThread1); semaphore.moveToThread(&consumerThread2); semaphore.moveToThread(&consumerThread3); // 启动线程 producerThread.start(); consumerThread1.start(); consumerThread2.start(); consumerThread3.start(); return a.exec(); } ``` 在上面的示例代码中,我们创建了一个QSemaphore对象作为信号量,并使用`acquire()`和`release()`函数来获取和释放信号量的锁定。生产者线程通过`acquire()`函数获取信号量的锁定,如果信号量的计数器为0,则线程会阻塞等待信号量的释放。消费者线程也使用相同的方法来获取和释放信号量的锁定。 这种方式实现了一个生产者多个消费者的场景,多个消费者线程可以在信号量的控制下轮流进行产品的消费。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值