Qt中事件分发源代码剖析

Qt中事件传递顺序:

在一个应该程序中,会进入一个事件循环,接受系统产生的事件,并且进行分发,这些都是在exec中进行的。
下面举例说明:

1)首先看看下面一段示例代码:

  1. int main(int argc, char *argv[])
  2. {
  3. QApplication a(argc, argv);
  4. MouseEvent w;
  5. w.show();
  6. return a.exec();
  7. }

2)a.exec进入事件循环,调用的是QApplication::exec();

  1. int QApplication::exec()
  2. {
  3. return <span style="color:#ff6666;">QGuiApplication::exec();</span>
  4. }

3)QApplication::exec()调用的是QGuiApplication::exec();

  1. int QGuiApplication::exec()
  2. {
  3. #ifndef QT_NO_ACCESSIBILITY
  4. QAccessible::setRootObject(qApp);
  5. #endif
  6. return QCoreApplication::exec();
  7. }

4)QGuiApplication::exec()调用的是QCoreApplication::exec();

  1. int QCoreApplication::exec()
  2. {
  3. if (!QCoreApplicationPrivate::checkInstance("exec"))
  4. return -1;
  5. QThreadData *threadData = self->d_func()->threadData;
  6. if (threadData != QThreadData::current()) {
  7. qWarning("%s::exec: Must be called from the main thread", self->metaObject()->className());
  8. return -1;
  9. }
  10. if (!threadData->eventLoops.isEmpty()) {
  11. qWarning("QCoreApplication::exec: The event loop is already running");
  12. return -1;
  13. }
  14. threadData->quitNow = false;
  15. QEventLoop eventLoop;
  16. self->d_func()->in_exec = true;
  17. self->d_func()->aboutToQuitEmitted = false;
  18. int returnCode = eventLoop.exec();
  19. threadData->quitNow = false;
  20. if (self) {
  21. self->d_func()->in_exec = false;
  22. if (!self->d_func()->aboutToQuitEmitted)
  23. emit self->aboutToQuit(QPrivateSignal());
  24. self->d_func()->aboutToQuitEmitted = true;
  25. sendPostedEvents(0, QEvent::DeferredDelete);
  26. }
  27. return returnCode;
  28. }

5)QCoreApplication::exec()调用eventLoop.exec()进行事件循环;

  1. int QEventLoop::exec(ProcessEventsFlags flags)
  2. {
  3. Q_D(QEventLoop);
  4. //we need to protect from race condition with QThread::exit
  5. QMutexLocker locker(&static_cast<QThreadPrivate *>(QObjectPrivate::get(d->threadData->thread))->mutex);
  6. if (d->threadData->quitNow)
  7. return -1;
  8. if (d->inExec) {
  9. qWarning("QEventLoop::exec: instance %p has already called exec()", this);
  10. return -1;
  11. }
  12. struct LoopReference {
  13. QEventLoopPrivate *d;
  14. QMutexLocker &locker;
  15. bool exceptionCaught;
  16. LoopReference(QEventLoopPrivate *d, QMutexLocker &locker) : d(d), locker(locker), exceptionCaught(true)
  17. {
  18. d->inExec = true;
  19. d->exit = false;
  20. ++d->threadData->loopLevel;
  21. d->threadData->eventLoops.push(d->q_func());
  22. locker.unlock();
  23. }
  24. ~LoopReference()
  25. {
  26. if (exceptionCaught) {
  27. qWarning("Qt has caught an exception thrown from an event handler. Throwing\n"
  28. "exceptions from an event handler is not supported in Qt. You must\n"
  29. "reimplement QApplication::notify() and catch all exceptions there.\n");
  30. }
  31. locker.relock();
  32. QEventLoop *eventLoop = d->threadData->eventLoops.pop();
  33. Q_ASSERT_X(eventLoop == d->q_func(), "QEventLoop::exec()", "internal error");
  34. Q_UNUSED(eventLoop); // --release warning
  35. d->inExec = false;
  36. --d->threadData->loopLevel;
  37. }
  38. };
  39. LoopReference ref(d, locker);
  40. // remove posted quit events when entering a new event loop
  41. QCoreApplication *app = QCoreApplication::instance();
  42. if (app && app->thread() == thread())
  43. QCoreApplication::removePostedEvents(app, QEvent::Quit);
  44. while (!d->exit)
  45. processEvents(flags | WaitForMoreEvents | EventLoopExec);
  46. ref.exceptionCaught = false;
  47. return d->returnCode;
  48. }

6)eventLoop.exec()调用QCoreApplication的processEvents进行事件分发;

7)调用notify进行分发

QCoreApplication::sendEvent、QCoreApplication::postEvent和QCoreApplication::sendPostedEvents都调用notify进行事件分发;

  1. bool QCoreApplication::notify(QObject *receiver, QEvent *event)
  2. {
  3. Q_D(QCoreApplication);
  4. // no events are delivered after ~QCoreApplication() has started
  5. if (QCoreApplicationPrivate::is_app_closing)
  6. return true;
  7. if (receiver == 0) {                        // serious error
  8. qWarning("QCoreApplication::notify: Unexpected null receiver");
  9. return true;
  10. }
  11. #ifndef QT_NO_DEBUG
  12. d->checkReceiverThread(receiver);
  13. #endif
  14. return receiver->isWidgetType() ? false :<span style="color:#ff6666;"> d->notify_helper</span>(receiver, event);
  15. }

8)notify调用notify_helper进行事件分发;

  1. bool QCoreApplicationPrivate::notify_helper(QObject *receiver, QEvent * event)
  2. {
  3. // send to all application event filters
  4. if (sendThroughApplicationEventFilters(receiver, event))
  5. return true;
  6. // send to all receiver event filters
  7. if (sendThroughObjectEventFilters(receiver, event))
  8. return true;
  9. // deliver the event
  10. return receiver->event(event);
  11. }

9)从上面第8步的代码可以看出事件传递

传递的顺序是:首先传递给全局的事件过滤器,再传递给目标对象的事件过滤器,最终传递给目标对象。

http://blog.csdn.net/chenlong12580/article/details/25009095

Qt中事件分发源代码剖析(一共8个步骤,顺序非常清楚:全局的事件过滤器,再传递给目标对象的事件过滤器,最终传递给目标对象)的更多相关文章

  1. Qt中事件分发源代码剖析

    Qt中事件分发源代码剖析 Qt中事件传递顺序: 在一个应该程序中,会进入一个事件循环,接受系统产生的事件,并且进行分发,这些都是在exec中进行的. 下面举例说明: 1)首先看看下面一段示例代码: i ...

  2. Qt中事件分发源码剖析

    Qt中事件分发源码剖析 Qt中事件传递顺序: 在一个应该程序中,会进入一个事件循环,接受系统产生的事件,而且进行分发,这些都是在exec中进行的. 以下举例说明: 1)首先看看以下一段演示样例代码: ...

  3. 关于JAVA中事件分发和监听机制实现的代码实例-绝对原创实用

    http://blog.csdn.net/5iasp/article/details/37054171 文章标题:关于JAVA中事件分发和监听机制实现的代码实例 文章地址: http://blog.c ...

  4. Android与javascript中事件分发机制的简单比较

    在前面两篇博客中,我们讨论了Android中的事件分发的相关内容,那么在本篇博客当中,我们就简单探讨一下html或javascript中的事件分发机制,并进行简单的对比. 在前端中,对事件进行绑定有三 ...

  5. QT中事件处理器和事件过滤器实现实例

    Qt中事件处理的方式,最常用的就是使用事件处理器(event handler)和事件过滤器(event filter)这两种方法.接下来,我们就来看看事件处理器和事件过滤器是怎么使用的. 事件处理器 ...

  6. JDK1.7中的ThreadPoolExecutor源代码剖析

    JDK1. 7中的ThreadPoolExecutor 线程池,顾名思义一个线程的池子,池子里存放了非常多能够复用的线程,假设不用线程池相似的容器,每当我们须要创建新的线程时都须要去new Threa ...

  7. Qt 中事件与处理

    一.事件与处理程序在运算过程中发生的一些事情:鼠标单击.键盘的按下...这些的事件的监控与处理在Qt中不是以信号的方式处理的.当这些事件发生时会调用QObject类中的功能函数(虚函数),所有的控件类 ...

  8. Android事件分发机制浅谈(二)--源码分析(ViewGroup篇)

    上节我们大致了解了事件分发机制的内容,大概流程,这一节来分析下事件分发的源代码. 我们先来分析ViewGroup中dispatchTouchEvent()中的源码 public boolean dis ...

  9. 第39课 Qt中的事件处理(下)

    1. 事件的传递过程 (1)操作系统检测到用户动作时,会产生一条系统消息,该消息被发送到Qt应用程序 (2)Qt应用程序收到系统消息后,将其转化为一个对应的QEvent事件对象,并调用QObject: ...

随机推荐

  1. python 文件中的中文编码解决方法

    # -*- coding: utf-8 -*- #查看安装的SDK默认的编码字符集在脚本中可以修改你的编码格式, 方法如下:#sys.getdefaultencoding()#reload(sys)# ...

  2. Eclipse vim插件安装使用

    在eclipse移动关闭位置感觉非常不爽,经常要用到方向键和鼠标,导致经常要移来移去.果断受不了了,去网上搜了下发现eclipse有许多vim插件可以使用.有一个大家都比较推荐的是 vrapper   ...

  3. mac OSX上eclipse adb无法识别(调试)小米的解决方案

    最近在Mac上开发安卓,用小米2a作为开发机,连上电脑后发现idea和eclipse真机调试的时候都提示USB device not found.经过一番google和百度,终于找到了解决方案,在这里 ...

  4. MySQL数据库配置主从服务器实现双机热备

    转自:http://www.cnblogs.com/cchun/p/3712637.html 一.安装MySQL 说明:在两台MySQL服务器192.168.21.169和192.168.21.168 ...

  5. hdu3870-Catch the Theves(平面图最小割)

    Problem Description A group of thieves is approaching a museum in the country of zjsxzy,now they are ...

  6. hdu4453-Looploop(伸展树)

    题目有很多图,不好粘贴..... 题意:给出N个数和K1,K2的值,最开始指针指向第一个数,有6种操作 add x : 给前K2个数都增加x reverse : 翻转前K1个数 insert x : ...

  7. poj 2697 A Board Game(bfs+hash)

    Description Dao was a simple two-player board game designed by Jeff Pickering and Ben van Buskirk at ...

  8. 数据库中简单的增删改查(CRUD)

    一切都是基于数据,而对数据的管理都离不开数据库.最近学到数据库的简单操作,所以写下这篇文章,总结一下学习到的知识.浅陋之处,多多见谅. 补充一下:一直弄不清SQL Server,Mysql ,以及Or ...

  9. 如何在编译内核时添加缺少的固件(随着intel wireless 5100 AGN的 iwlwifi 案例)

    我不知道你在笔记本使用 Linux 在内核编译无线wifi 不能用.我的书"关联 Y450"一个足够的旧书,随着无线网卡: $ lspci | grep Wireless 06:0 ...

  10. oracle 10g 数据库字符集更改

    1.更改数据库字符集为GBK SHUTDOWN IMMEDIATE; STARTUP MOUNT EXCLUSIVE; ALTER SYSTEM ENABLE RESTRICTED SESSION;A ...