Qt QEventLoop的使用的一个问题讨论

先看一段代码

cpp 复制代码
int loop=0;
void xxx()
{
    int nIndex = loop++;
qDebug()<<"a:"<<nIndex;
    //构建一个eventLoop,来阻塞3s
    QEventLoop eventLoop;
    QTimer::singleShot(3000, [&eventLoop](){
        eventLoop.quit();
    });
    eventLoop.exec();
qDebug()<<"b:"<<nIndex;
}
int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);
    QTimer::singleShot(1000,[]{
        for(int i=0;i<5;i++)
        {
            xxx();
            QThread::msleep(30);
        }
    });
    qDebug()<<"before application exec";
    return a.exec();
}

这段代码在xxx()函数中用定时器模拟了一个长时间执行的任务,该任务通过放在QEventLoop后面,开启了事件循环来让任务完成后继续执行。

再看另外一段代码

cpp 复制代码
MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{
    QPushButton* btn=new QPushButton(this);
    this->setCentralWidget(btn);

    connect(btn,&QPushButton::clicked,this,&MainWindow::xxx);
}



void MainWindow::xxx()
{
    //当前调用顺序值
    int nIndex = m_nCurIndex++;
qDebug()<<"a:"<<nIndex;
    //构建一个eventLoop,来阻塞3s
    QEventLoop eventLoop;
    QTimer::singleShot(3000, [&eventLoop](){
        eventLoop.quit();
    });
    eventLoop.exec();
qDebug()<<"b:"<<nIndex;
}

第二段代码和第一段的区别在于,同样是在xxx()中开启了耗时的任务,并且放在事件循环之后。

此时在界面上用鼠标连续点击三次,看看程序输出。

这是第一段代码的输出

before application exec

a: 0

b: 0

a: 1

b: 1

a: 2

b: 2

a: 3

b: 3

a: 4

b: 4

下面是第二段代码的输出

a: 0

a: 1

a: 2

b: 2

b: 1

b: 0

先思考一下为何有这样的区别。

先说第一段,第一段代码确确实实事件循环阻塞了,它使得五次调用xxx()都是依次调用的。每一次都是一个xxx()函数执行完毕后,进入下一个xxx()

但是第二段,情况有所不一样。当点击第一次按钮,调用xxx()时,会阻塞在eventLoop.exec();

**但是用户的界面点击操作并没有被阻塞,**用户点击第二次时,信号槽连接的情况下,会第二次进入xxx(),又会阻塞在eventLoop.exec();以及继续点击第三次后,继续进入xxx()阻塞在eventLoop.exec();

所以三次点击后,程序三次阻塞在eventLoop.exec();

之后,三次定时器依次超时,按理说会依次输出b:0 b:1 b:2

但实际情况是反过来的。

需要注意,第二段代码中的三次xxx()是在同一个线程中执行的,它们依据先后顺序在同一个栈里面,所以有了先进的后出,输出结果序列反过来的问题。

考虑第二个问题,当一次点击后进入xxx()阻塞在evenLoop.exec()后,禁止用户点击的第二次立刻相应,否则造成多次重叠,并且最后的执行顺序还是反过来的。

那么使用eventLoop.exec(QEventLoop::ExcludeUserInputEvents);来禁止用户连续的操作

此时,多次连续点击后,第二段代码的输出如下--顺序上已经保证了一致。

a: 0

b: 0

a: 1

b: 1

a: 2

b: 2

相关推荐
逍遥德4 小时前
Sring事务详解之02.如何使用编程式事务?
java·服务器·数据库·后端·sql·spring
笨蛋不要掉眼泪4 小时前
Redis哨兵机制全解析:原理、配置与实战故障转移演示
java·数据库·redis·缓存·bootstrap
子春一4 小时前
Flutter for OpenHarmony:构建一个 Flutter 数字消消乐游戏,深入解析网格状态管理、合并算法与重力系统
算法·flutter·游戏
小杨同学呀呀呀呀4 小时前
Ant Design Vue <a-timeline>时间轴组件失效解决方案
前端·javascript·vue.js·typescript·anti-design-vue
Coder_Boy_4 小时前
基于SpringAI的在线考试系统-整体架构优化设计方案
java·数据库·人工智能·spring boot·架构·ddd
草履虫建模10 小时前
力扣算法 1768. 交替合并字符串
java·开发语言·算法·leetcode·职场和发展·idea·基础
华玥作者12 小时前
[特殊字符] VitePress 对接 Algolia AI 问答(DocSearch + AI Search)完整实战(下)
前端·人工智能·ai
naruto_lnq13 小时前
分布式系统安全通信
开发语言·c++·算法
Mr Xu_13 小时前
告别冗长 switch-case:Vue 项目中基于映射表的优雅路由数据匹配方案
前端·javascript·vue.js
Jasmine_llq13 小时前
《P3157 [CQOI2011] 动态逆序对》
算法·cdq 分治·动态问题静态化+双向偏序统计·树状数组(高效统计元素大小关系·排序算法(预处理偏序和时间戳)·前缀和(合并单个贡献为总逆序对·动态问题静态化