• 【QT小记】QT线程同步--QWaitCondition


    QWaitCondition提供了一个同步线程的条件变量

    主要函数

    阻塞线程,等待被唤醒

    • 在执行wait之前需先获取lockedMutex
      (wait函数底层执行对lockedMutex的处理:在执行wait阻塞线程后,会释放lockedMutex;在唤醒线程后,会重新获取lockedMutex)

    bool wait(QMutex *lockedMutex, unsigned long time = ULONG_MAX)

    唤醒所有在等待条件下的线程,但唤醒顺序未知,依赖系统调度策略
    void wakeAll()

    随机唤醒一个在等待条件下的线程,具体唤醒哪一个线程,依赖系统调度策略
    void wakeOne()

    简单的使用例子
    QWaitCondition waitCondition;
    QMutex mutex;
    
    void Widget::on_btnWait_clicked()
    {
        auto threadFun = [=] {
            for (int i = 0; i < 10; ++i) {
                qDebug() << "i = " << i;
                if (7 == i) {
                    qDebug() << "i == 7时阻塞线程,待唤醒";
                    mutex.lock();  // 阻塞线程前获取锁
                    waitCondition.wait(&mutex);
                    mutex.unlock();
                }
            }
        };
    
        QtConcurrent::run(threadFun);
    }
    
    void Widget::on_btnWake_clicked()
    {
        qDebug() << "唤醒线程";
        waitCondition.wakeAll();
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    单生产者单消费者模型

    QT官方使用QWaitCondition实现的例子
    https://code.qt.io/cgit/qt/qtbase.git/tree/examples/corelib/threads/waitconditions/waitconditions.cpp?h=5.14

    • 有一个存放数据的容器buffer
    • 生产者生产数据存放于buffer,消费者从buffer中取出数据消费
    • 消费者在从buffer中获取数据时,若buffer为空,则阻塞消费者,待生产者生产数据(buffer中有数据即可)后唤醒消费者来获取数据
    • 生产者在生产数据存放于buffer时,若buffer已满,则阻塞生产者,待消费者取出数据(buffer中有空闲位置即可)后唤醒生产者来生产数据
    // QT官方Demo
    const int DataSize = 100000;   //数据总量
    
    const int BufferSize = 8192;   //容器大小
    char buffer[BufferSize];       //容器
    
    QWaitCondition bufferNotEmpty; //条件变量: 容器不为空
    QWaitCondition bufferNotFull;  //条件变量: 容器未满
    QMutex mutex;                  //锁
    int numUsedBytes = 0;
    
    // 生产者线程
    class Producer : public QThread
    {
    public:
        Producer(QObject *parent = NULL) : QThread(parent)
        {
        }
    
        void run() override
        {
            for (int i = 0; i < DataSize; ++i) {
                mutex.lock();
                // 若容器已满,则阻塞生产者线程,待容器未满时唤醒
                if (numUsedBytes == BufferSize)
                    bufferNotFull.wait(&mutex);
                mutex.unlock();
                
                // 生产数据
                buffer[i % BufferSize] = "ACGT"[QRandomGenerator::global()->bounded(4)];
    
                mutex.lock();
                ++numUsedBytes;
                // 生产数据后则容器不为空,则可唤醒消费者线程
                bufferNotEmpty.wakeAll();
                mutex.unlock();
            }
        }
    };
    
    // 消费者线程
    class Consumer : public QThread
    {
    public:
        Consumer(QObject *parent = NULL) : QThread(parent)
        {
        }
    
        void run() override
        {
            for (int i = 0; i < DataSize; ++i) {
                mutex.lock();
                // 若容器为空,则阻塞消费者线程,待容器不为空时唤醒
                if (numUsedBytes == 0)
                    bufferNotEmpty.wait(&mutex);
                mutex.unlock();
    
                fprintf(stderr, "%c", buffer[i % BufferSize]);
    
                mutex.lock();
                --numUsedBytes;
                // 取出一个数据后则容器未满,则可唤醒生产者线程
                bufferNotFull.wakeAll();
                mutex.unlock();
            }
            fprintf(stderr, "\n");
        }
    };
    
    int main(int argc, char *argv[])
    {
        QCoreApplication app(argc, argv);
        Producer producer;
        Consumer consumer;
        producer.start();
        consumer.start();
        producer.wait();
        consumer.wait();
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80

    若把容器的大小设置为1,则可清晰的看到程序的执行流程是按生产者线程生产一个数据,消费者线程消费一个数据的顺序执行。

    学习资料

    QT帮助文档

  • 相关阅读:
    排序-表排序
    django--->自定义表名,建立索引
    Nacos 高级玩法:深入探讨分布式配置和服务发现
    37. 解数独
    DOM4J解析.XML文件
    为什么您的公司应该进行脉动调查以及如何正确进行
    治理污染要从有效防护开始,PM2.5在线监测系统
    YOLOv5论文作图教程(2)— 软件界面布局和基础功能介绍
    基于Java实现的植物大战僵尸游戏
    Windows11右键菜单修改为Win10模式的方法
  • 原文地址:https://blog.csdn.net/lin786063594/article/details/126679860