Another Look at Events(再谈Events)
转载:http://www.qtcn.org/bbs/simple/?t31383.html
Another Look at Events(再谈Events)
最近在学习Qt事件处理的时候发现一篇很不错的文章,是2004年季刊的一篇文章,网上有这篇文章的翻译版,但是感觉部分地方翻译的比较粗糙,不是很明确。索性重新翻译了一遍,并引用了原翻译版的一段译注。以下都是用自己能理解的方式来翻译的,由于水平有限,有很多不足的地方,希望大家指正。
Another Look at Events (再谈Events)
by Jasmin Blanchette
什么是自发事件?哪些事件类型可以被传递或压缩到一起?投递(posting)和发射(sending)一个事件的不同点在哪里?我应该什么时候在一个事件中调用accept()或ignore()?如果你不知道前面所有的问题,那么请继续阅读下文!
• 事件的来源
• 手工事件
• 自定义事件类型
• 事件处接受或忽略?
事件的来源
事件可以基于产生和发送的方式来分为三种类型:
• 自发事件(spontaneous events)由窗口系统所产生。它们被放入一个系统队列,并通过事件循环相继地进行处理
• 投递事件(posted events)由Qt应用程序所产生。它们被放入Qt的消息队列中等待,并通过事件循环进行处理。
• 发射事件(sent events)由Qt应用程序所产生,但是它们被直接发送给目标对象。
当我们在main()函数的最后调用QApplication::exec()时,应用程序就进入Qt事件循环。从概念上讲,事件循环像如下所示的方式:
while (!exit_was_called) {
while (!posted_event_queue_is_empty) {
process_next_posted_event();
}
while (!spontaneous_event_queue_is_empty) {
process_next_spontaneous_event();
}
while (!posted_event_queue_is_empty) {
process_next_posted_event();
}
}
首先,事件循环处理所有的投递事件(posted events),直到事件队列为空。然后,它处理自发事件(spontaneous events),直到处理完所有事件。最后,它处理所有在处理自发事件(spontaneous events)过程中所产生的投递事件(posted events)。
发射事件(sent events)不会被事件循环所处理。它们被直接传递给对象。
让我们看看这在绘制事件(paint events)的执行中是如何工作的。当一个窗口小部件(widget)第一次显示时,或者当它被隐藏后又变的可见时,窗口系统会产生一个(自发的)要求程序重绘该窗口小部件的绘制事件(paint event)。最后,事件循环获取该事件,并将它发送给需要重绘的窗口小部件。
不是所有的绘制事件(paint events)都是由窗口系统所产生。当你调用QWidget::update()来重绘一个窗口小部件时,该窗口小部件向自身投递一个绘制事件(paint event)。该绘制事件排队等待,并在最后由事件循环分派。
如果你没有耐心,并且不想等待事件循环去重绘一个窗口小部件,那么理论上,你可以直接调用paintEvent()来强制立即重绘。但是在实际中,这样做并不一定可行,因为paintEvent()是一个受保护的函数。这也将绕过任何现有的事件过滤器。基于这一原因,Qt提供了一个直接发射(sending)事件,而不是投递(posting)事件的机制。QWidget::repaint()使用这一机制来强制进行立即的重绘。
投递事件(posting events)相比于发射(sending)事件的一个优势在于投递(posting)的方式使Qt有机会对事件进行压缩。如果你在同一个窗口小部件(widget)上连续地调用10次update(),并且没有返回事件循环(event loop),那么由update()所产生的10次事件将自动被合并为一个单一的事件,这一过程伴随着在它们所有的QPaintEvents对象中指定的区域的合并。
最后,要注意的是你可以在任何时候调用QApplication::sendPostedEvents()以强制Qt处理一个对象的投递事件(posted events)。
手工事件(即由应用程序主动激发,而不是由外界激发的事件)
Qt应用程序可以产生它们自己的预定义或自定义类型的事件。这是通过创建一个QEvent(或其子类)的对象实例,并调用QApplication::postEvent() 或 QApplication::sendEvent()来实现的。
这两个函数都有一个QObject *和一个QEvent *类型的参数。如果你调用postEvent(),那么你必须使用new关键字来创建一个事件对象(event object),并且当它被处理完以后,Qt将自动删除它。如果你调用sendEvent(),那么你必须在栈(stack)上创建事件对象。这里有一个投递(posting)一个事件的例子:
QApplication::postEvent(mainWin, new QKeyEvent(QEvent::KeyPress, Key_X, 'X', 0));
这里是一个发射(sending)一个事件的例子:
QKeyEvent event(QEvent::KeyPress, Key_X, 'X', 0);
QApplication::sendEvent(mainWin, &event);
Qt应用程序很少需要直接调用postEvent() 或 sendEvent(),因为大多数事件在需要时会由Qt或窗口系统自动产生。在大多数你想发送一个事件的情形中,Qt包含了一个高级的函数来代替它(例如update() 和repaint())。
自定义事件类型
Qt使你可以创建自己的事件类型。这一技术在多线程应用程序中特别有用,可以作为一种与GUI线程通信的手段;请参阅C++ GUI Programming with Qt 3 (p. 359)第17章的一个例子。
自定义类型在单线程应用程序中也是很有用的,可以作为一种对象间的通信机制。你愿意使用事件而不是标准函数调用或信号与槽的主要原因是事件既可以同步使用,也可以异步使用(取决于你是否调用sendEvent()或postEvents()),然而调用一个函数或一个槽就总是同步的。事件的另一个优势是他们可以被过滤。关于这一点更详细的论述在下面的一节。
这里有一个展示如何投递(post)一个自定义事件的代码片段:
const QEvent::Type MyEvent = (QEvent::Type)1234;
...
QApplication::postEvent(obj, new QCustomEvent(MyEvent));
事件必须是QCustomEvent(或一个子类)的类型。构造函数的参数是事件的类型。1024以下的值被Qt所保留用于预定义的事件类型;其它可以被应用程序所使用。
(译者注:在Qt4中,自定义事件是通过子类化QEvent来创建的。事件特定的数据被按照适合你的应用程序的方式存储。自定义事件仍然被发送给每个对象的customEvent()处理器函数,但是它们被作为QEvent对象而不是已经过时的QCustomEvent对象处理)
为了处理自定义事件类型,需要重载customEvent()函数:
void MyLineEdit::customEvent(QCustomEvent *event)
{
if (event->type() == MyEvent) {
myEvent();
} else {
QLineEdit::customEvent(event);
}
}
QCustomEvent拥有一个void *类型的成员,你可以将其用于自己的目的。如果你想要更多的类型安全,那么你也可以子类化QCustomEvent,并添加其它成员。但是然后你也需要在customEvent()中将QCustomEvent对象指向你特定的类型。
事件处理和过滤
Qt中的事件可以进行5个不同层次的处理:
• 重新实现一个指定的事件处理器。
QObject 和 QWidget为不同的事件类型提供了许多特殊的事件处理器(例如,用于处理绘制事件的paintEvent())。
• 重载QObject::event()。
event()函数是一个对象的所有事件的入口点。QObject 和 QWidget中的该函数的默认实现只是简单地将事件转发给指定的事件处理器。
• 在一个QObject对象中安装一个事件过滤器。事件过滤器是一种对象,它能在另一个对象的事件到达指定目标之前,接收这些事件。
• 在qApp中安装一个事件过滤器。特别地,一个qApp上的事件过滤器监控所有的发送给应用程序中全部对象的事件。
• 重载QApplication::notify()。Qt事件循环和sendEvent()调用该函数以分发事件。通过重载它,你可以在其它任何人之前查看事件。
一些事件类型可以被蔓延传播。这意味着如果一个目标对象不处理某个事件,那么Qt设法为事件找到另外一个接收者,并用新的目标对象调用QApplication::notify()。
例如,按键事件被传播;如果拥有焦点的窗口小部件(widget)不处理某一个按键事件,那么Qt会向其父窗口小部件发送一个相同的事件,然后再发送给父窗口小部件的父窗口等等,直到事件到达最顶层的窗口小部件。
接受或忽略?
能被传播的事件都拥有一个accept()和一个ignore()函数,你可以调用它们来告诉Qt系统你“接受”或“忽略”该事件。如果一个事件处理器在某个事件上调用accept(),那么该事件就不会被继续传播;如果事件处理器是调用ignore(),那么Qt会设法寻找另一个接收者。
如果你和大多数Qt开发者一样,那么你可能从不会觉得在你的程序中调用accept()和ignore()是一件麻烦事。正是这样,Qt是以一种你通常从不需要调用这些函数的方式来进行设计的。默认的值是“accept”,而默认的在QWidget中的事件处理器的执行会调用ignore()。如果你想接受一个事件,那么你仅仅需要重载事件处理器,并且避免调用QWidget的事件处理器。如果你想忽略一个事件,那么只需要简单地将其传递给QWidget的事件处理器。下面的代码片段说明了这一点:
void MyFancyWidget::keyPressEvent(QKeyEvent *event)
{
if (event->key() == Key_Escape) {
doEscape();
} else {
QWidget::keyPressEvent(event);
}
}
在这个例子中,如果用户按下了Esc键,我们就调用doEscape(),并且事件被接受(默认)。该事件不会被继续传播给父窗口小部件。如果用户按下了其他任何一个按键,那么我们调用QWidget的默认事件处理器。
void QWidget::keyPressEvent(QKeyEvent *event)
{
event->ignore();
}
由于调用了ignore(),事件将被传播给父窗口小部件。
到目前为止,我们都假设QWidget是基类。但是,通过将QWidget替换为其它基类,同样的用法可以在任何层次上使用。例如:
void MyFancyLineEdit::keyPressEvent(QKeyEvent *event)
{
if (event->key() == Key_SysReq) {
doSystemRequest();
} else {
QLineEdit::keyPressEvent(event);
}
}
如果由于某些原因,你使用event()函数代替特定的事件处理器(例如keyPressEvent())来处理事件,那么程序就稍微有点不同。event()函数会返回一个布尔值来告诉调用者某个事件是否被接受或忽略(true表示“接受”)。在event()中调用一个事件上的accept()或ignore()是没有意义的。“accept”标记是特定的事件处理器和event()之间的一种通信机制,而event()返回的布尔值被用于同QApplication::notify()的通信。QWidget中默认的event()将“accept”标识转换为一个布尔值的执行过程如下所示:
bool QWidget::event(QEvent *event)
{
switch (e->type()) {
case QEvent::KeyPress:
keyPressEvent((QKeyEvent *)event);
if (!((QKeyEvent *)event)->isAccepted())
return false;
break;
case QEvent::KeyRelease:
keyReleaseEvent((QKeyEvent *)event);
if (!((QKeyEvent *)event)->isAccepted())
return false;
break;
...
}
return true;
}
到目前为止所讨论的内容不仅适用于按键事件,也适用于鼠标、滚轮、手写板和上下文菜单(即右键菜单)事件。
关闭事件的工作方式有所不同。调用QCloseEvent::ignore()可以取消关闭操作,但是通常accept()告诉Qt继续执行关闭操作。为了避免任何的混乱,在你重载后的closeEvent()中明确地调用accept()或ignore()是一个好办法,就像这样:
void MainWindow::closeEvent(QCloseEvent *event)
{
if (userReallyWantsToQuit()) {
event->accept();
} else {
event->ignore();
}
}
Another Look at Events(再谈Events)的更多相关文章
- 【一次面试】再谈javascript中的继承
前言 面向对象编程是每次面试必问的知识点,而前端js如何实现继承每次命中率高达80% 这不,近两天我们面试时候,同事就问道面试者此问题,但是,不论之前自己做的回答,还是面试者的回答,基本都不太令人满意 ...
- [转载]再谈百度:KPI、无人机,以及一个必须给父母看的案例
[转载]再谈百度:KPI.无人机,以及一个必须给父母看的案例 发表于 2016-03-15 | 0 Comments | 阅读次数 33 原文: 再谈百度:KPI.无人机,以及一个必须 ...
- Support Vector Machine (3) : 再谈泛化误差(Generalization Error)
目录 Support Vector Machine (1) : 简单SVM原理 Support Vector Machine (2) : Sequential Minimal Optimization ...
- Unity教程之再谈Unity中的优化技术
这是从 Unity教程之再谈Unity中的优化技术 这篇文章里提取出来的一部分,这篇文章让我学到了挺多可能我应该知道却还没知道的知识,写的挺好的 优化几何体 这一步主要是为了针对性能瓶颈中的”顶点 ...
- 浅谈HTTP中Get与Post的区别/HTTP协议与HTML表单(再谈GET与POST的区别)
HTTP协议与HTML表单(再谈GET与POST的区别) GET方式在request-line中传送数据:POST方式在request-line及request-body中均可以传送数据. http: ...
- C++ Primer 学习笔记_32_STL实践与分析(6) --再谈string类型(下)
STL实践与分析 --再谈string类型(下) 四.string类型的查找操作 string类型提供了6种查找函数,每种函数以不同形式的find命名.这些操作所有返回string::size_typ ...
- 再谈JSON -json定义及数据类型
再谈json 近期在项目中使用到了highcharts ,highstock做了一些统计分析.使用jQuery ajax那就不得不使用json, 可是在使用过程中也出现了非常多的疑惑,比方说,什么情况 ...
- C++ Primer 学习笔记_44_STL实践与分析(18)--再谈迭代器【下】
STL实践与分析 --再谈迭代器[下] 三.反向迭代器[续:习题] //P355 习题11.19 int main() { vector<int> iVec; for (vector< ...
- C++ Primer 学习笔记_43_STL实践与分析(17)--再谈迭代器【中】
STL实践与分析 --再谈迭代器[中] 二.iostream迭代[续] 3.ostream_iterator对象和ostream_iterator对象的使用 能够使用ostream_iterator对 ...
随机推荐
- Jenkins 初见
在网上貌似没有找到Jenkins的中文的太多的文档,有的都是关于Hudson的一些 零零散散的,所以自己边学习边实践总结了以下系列文章,希望有助于大家对于Jenkins的使用. 本系列文章是基于我3年 ...
- 在SQL中用正则表达式替换html标签
由于数据库的一个表字段中多包含html标签,现在需要修改数据库的字段把html标签都替换掉.当然我可以通过写一个程序去修改,那毕竟有点麻烦.直接在查询分析器中执行,但是MS SQL Server并没有 ...
- DTN学习的一些有用链接
1.DTN研究组,该网站提供了一些代码,有NS2上的实现,也有用java实现的源码. http://www.dtnrg.org/wiki/Code 2.DTN实现的另一个版本,与ONE比较,目前还没用 ...
- [Angular 2] A Simple Form in Angular 2
When you create a Form in Angular 2, you can easily get all the values from the Form using ControlGr ...
- hdu1074 Doing Homework(状态压缩DP Y=Y)
Doing Homework Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) T ...
- .NET简单的语句
获取当前时间的代码: Response.Write(DateTime.Now); 第一次加载页面的语句: if (!IsPostBack) { Response.Write("这是第一次加载 ...
- linux lvm的操作手册_pvcreate_vgcreate_lvcreate_相关
一. 前言 每个Linux使用者在安装Linux时都会遇到这样的困境:在为系统分区时,如何精确评估和分配各个硬盘分区的容量,因为系统管理员不但要考虑到当前某 个分区需要的容量,还要预见该分区以后可能需 ...
- 简单题思维转化BestCoder
题意:给你a, b, c, d四个数,这几个数的范围都是大于0小于1000的整数,让比较 a ^b 和 c ^ d的大小. 这道题看着特别简单,但是当时就是做不出来,将近一个月没有做题了,手生了,不过 ...
- tomcat startup.sh提示java.lang.OutOfMemoryError: PermGen space
JAVA_OPTS="-server -XX:PermSize=512M -XX:MaxPermSize=1024m"if [ -z "$LOGGING_MANAGER& ...
- 关于oracle dbms_job 定时执行的内容。
一.设置初始化参数 job_queue_processessql> alter system set job_queue_processes=n;(n>0)job_queue_proces ...