Qt-事件循环

事件循环

是什么

所谓事件,可以大致分为一下几类:

  • 键盘、鼠标以及其他与窗体交互引发的事件。
  • socket活动,如连接、可读、可写引发的事件
  • 定时器超时引发的事件
  • 从其他线程中手动发出的事件
    事件生成后,并非立刻发送,而是放在事件队列(event queue)中,一定时间后发送。分发器(dispatcher)循环获取事件队列中的事件,并把事件发送至事件的目标对象,因此称之为事件循环。事件循环伪代码如下:
while (is_active)
{
    while (!event_queue_is_empty)
        dispatch_next_event();
    wait_for_more_events();
}

通过QCoreApplication::exec()函数,我们就进入到一个事件循环中,当调用QCoreApplication::exit()或者QCoreApplication::quit()函数,循环便终止了。

事件循环嵌套

一个Qt应用通常至少有一个事件循环,那就是main()里面调用的QCoreApplication::exec(),除此之外还可能有其他的事件循环,如通过QEventLoop::exec(),我们进入了一个新的事件循环,通常称之为本地事件循环,类似的方法还有QDialog::exec()、QMessageBox::exec()。
所谓嵌套,是指QEventLoop::exec()运行在QCoreApplication::exec()的事件循环中,那么此时UI会被冻结吗?不会。因为此时进入了本地事件循环,大部分父循环中的事件将在子循环中处理。如果父循环要退出,将等到子循环退出后才退出。

事件循环嵌套的应用

主线程等待

如果以sleep方式将冻结UI

QEventLoop loop;
QTimer::singleShot(100, &loop, SLOT(quit()));
loop.exec();

模拟模态对话框

QDialog dlg;
dlg.show();
QEventLoop loop;
connect(&dlg, SIGNAL(finished(int)), &loop, SLOT(quit()));
loop.exec(QEventLoop::ExcludeUserInputEvents);

同步获取数据

void A::onFinish(bool r, const QString &info)
{
   m_result = r;
   qDebug() << info;
 //Exit event loop in slot 
  loop.quit(); 
}

bool A::get(const QString &userName, const QString &passwdHash, const QString &dataName)
 { 
    //Declare local EventLoop QEventLoop loop;
    m_result = false; 
   //Connect the signal first 
   connect(&network, SIGNAL(finished(bool,const QString &)),this,SLOT(onFinish(bool,const QString &)));
 //Initiate login request 
  getData(userName, passwdHash, dataName); 
//Start the event loop. Block the current function call, but the event loop can still run. 
//This is not going to run down to the front slot, calling loop.. After quitting, we will continue to go down
 loop.exec();
 //Return result. Before loop exits, M_ The value in result has been updated.
 return m_result;
 }

主线程同步执行代码

QEventLoop loop;
QThread* pThread = QThread::create([&]()
{
	doSomething();
});
connect(pThread, &QThread::finished, &loop, &QEventLoop::quit);
pThread->start();
loop.exec();

不要阻塞事件循环

即不要在事件循环中执行耗时的操作,因为这将导致其他事件无法被处理,例如鼠标点击事件的响应槽函数中,不应做耗时操作,因为事件将无法被处理,UI会冻结,程序表现出“无响应”的状态。

线程的事件循环

QThread::exec()将开启一个新的事件循环,每个线程的事件循环独立,并且只为存在于(living)其线程内的QObjects对象传递事件,这些对象分为两类,一类是在其线程内创建的,一类是通过moveToThread()移动到该线程内的。可通过thread()方法查看对象依附的线程(thread affinity)。
对象、线程、事件循环的关系如下图:
在这里插入图片描述

跨线程信号槽

槽函数总在槽对象所在线程执行,信号连接有5种类型:

  • Qt::DirectConnection。信号发送时所在线程与槽对象所在线程相同,等价于在信号发送处替换为槽函数调用,不需要事件循环参与
  • Qt::QueuedConnection。信号发送时所在线程与槽对象线程不同,此时会发送一个事件到槽对象所在线程的事件队列
  • Qt::BlockingQueuedConnection。与Qt::QueuedConnection相同,但信号发送线程将阻塞直到槽函数执行并返回
  • Qt::AutoConnection。根据发送信号时所在线程与槽对象所在线程判断使用Qt::DirectConnection或Qt::QueuedConnection
  • Qt::UniqueConnection。与Auto Connection相同,不同的地方在于如果已经存在完全相同的连接,则不进行连接

deleteLater与事件循环

deleteLater()将发送一个QEvent::DeferredDelete事件到事件循环中。
deleteLater()什么时候真正删除对象?

  • 当控制返回事件循环时,该对象将被删除
  • 如果调用deleteLater()时事件循环未运行,一旦事件循环开始,该对象将被删除
  • 如果在循环停止后调用deleteLater(),该对象将在线程完成时被销毁(Qt 4.8以后)

注意,进入和离开新的事件循环(例如QDialog::exec())将不会执行延迟删除,控件权必须返回到调用deleteLater()时所在的事件循环时,对象才会被删除。

参考:

  1. https://wiki.qt.io/Threads_Events_QObjects
  2. https://programming.vip/docs/qt-event-loop-and-use-of-qeventloop.html
  3. https://stackoverflow.com/questions/50450912/how-local-qeventloop-works
  4. https://stackoverflow.com/questions/61110472/how-qt-handle-events-and-signal-in-same-eventloop
  5. https://forum.qt.io/topic/100334/nested-qeventloop
  6. https://www.codetd.com/en/article/6601268
  7. https://stackoverflow.com/questions/22376298/when-to-use-deletelater#:~:text=The%20object%20will%20be%20deleted,the%20event%20loop%20is%20started.
  • 3
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Qt事件循环是一个非常重要的概念,也是Qt能够实现异步、响应式编程的关键所在。简单来说,事件循环是一个无限循环,用于处理Qt应用程序中的各种事件,包括用户输入、定时器事件、网络事件等等。以下是Qt事件循环的原理: 1. 事件队列:Qt应用程序中的所有事件都被放置在一个事件队列中,事件队列是一个先进先出的队列结构。 2. 事件循环Qt应用程序启动后,会进入一个无限循环,也就是事件循环事件循环会不断地从事件队列中取出事件,并将事件分发给对应的对象进行处理。 3. 事件分发:当事件循环从事件队列中取出一个事件时,会根据事件的类型和目标对象,将事件分发给对应的对象进行处理。如果事件是一个用户输入事件,那么它会被分发给当前获得焦点的控件进行处理;如果事件是一个定时器事件,那么它会被分发给对应的定时器对象进行处理。 4. 事件处理:一旦事件被分发给对应的对象,该对象就会调用自己的事件处理函数来处理事件。例如,一个按钮控件的事件处理函数会检测用户是否点击了该按钮,如果是,则执行与该按钮相关的操作。 5. 事件过滤器:Qt框架允许对象在事件处理之前拦截事件进行处理,这个过程称为事件过滤。事件过滤器可以用来处理一些全局事件,例如窗口关闭事件、应用程序退出事件等等。 总之,Qt事件循环是一个非常灵活、高效的机制,可以保证Qt应用程序能够实现异步、响应式编程。开发者只需要关注对象的事件处理函数,无需关心事件循环的具体实现。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

mrbone11

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值