qobject_cast的类型转换问题

blueliuyun 2011-08-10 11:05:35
T qobject_cast ( QObject * object )
Returns the given object cast to type T if the object is of type T (or of a subclass); otherwise returns 0. If object is 0 then it will also return 0.

The class T must inherit (directly or indirectly) QObject and be declared with the Q_OBJECT macro.

A class is considered to inherit itself.

Example:

QObject *obj = new QTimer; // QTimer inherits QObject

QTimer *timer = qobject_cast<QTimer *>(obj);
// timer == (QObject *)obj

QAbstractButton *button = qobject_cast<QAbstractButton *>(obj);
// button == 0


Q:看了Assistant之后,仍有疑问,请朋友帮忙解答下吧,先谢过!
在Example中的第二个QAbstractButton *button为什么返回的
button == 0 呢?
有人说“转换的对象类型必须是obj的父类或者同级才能得到正确的结果,同时还得注意Q_OBJECT宏定义。”
那么是否可以认为 QObject <- QWidget <- QAbstractButton ;
QObject <- QTimer ; 由于QAbstractButton 与 QTimer不同级所以在Example中的第二个返回的
button == 0 ?
...全文
1154 7 打赏 收藏 转发到动态 举报
写回复
用AI写文章
7 条回复
切换为时间正序
请发表友善的回复…
发表回复
blueliuyun 2011-08-11
  • 打赏
  • 举报
回复
[Quote=引用 6 楼 lefttime 的回复:]
首先, 纠正个小问题, qobject_cast和是否重入无关哦 :)
向下转换(下行转换): 父类向子类进行转换,此时有个类型检查的过程, 像dynamic_cast一样;


C/C++ code

QObject *obj = new QTimer; // QTimer inherits QObject

QTimer* timer = qobject_ca……
[/Quote]
好的,我记住了,谢谢。
乔巴好萌 2011-08-10
  • 打赏
  • 举报
回复
assistant上写了
类型兼容的话 尝试转换才会成功 否则返回0
QTimer是QtCore的方法,抽象按钮是UI的方法,一个可重入,一个不可重入,不一样的
乔巴好萌 2011-08-10
  • 打赏
  • 举报
回复
不是的
你可以看下assistant上的说明
It attempts to cast its argument to the pointer type specified in angle-brackets, returning a non-zero pointer if the object is of the correct type (determined at run-time), or 0 if the object's type is incompatible.
lefttime 2011-08-10
  • 打赏
  • 举报
回复
qobject_cast转换的必须是继承于QObject的类, 向下转换, 类似于dynamic_cast,而例子中的obj生成的实例是QTimer, 自然第二个返回的button==0; 并不是因为不处在同一级而返回0, 这个要注意了!
lefttime 2011-08-10
  • 打赏
  • 举报
回复
首先, 纠正个小问题, qobject_cast和是否重入无关哦 :)
向下转换(下行转换): 父类向子类进行转换,此时有个类型检查的过程, 像dynamic_cast一样;


QObject *obj = new QTimer; // QTimer inherits QObject

QTimer* timer = qobject_cast<QTimer*>(obj); // 这里就是下行转换
// timer == (QObject *)obj

QAbstractButton* button = qobject_cast<QAbstractButton*>(obj);
// button == 0

QPushButton* buttonObj = new QPushButton;
button = qobject_cast<QAbstractButton*>(buttonObj); // 这里就是上行转换,不存在是否抽象之说
blueliuyun 2011-08-10
  • 打赏
  • 举报
回复
[Quote=引用 1 楼 lefttime 的回复:]
qobject_cast转换的必须是继承于QObject的类, 向下转换, 类似于dynamic_cast,而例子中的obj生成的实例是QTimer, 自然第二个返回的button==0; 并不是因为不处在同一级而返回0, 这个要注意了!
[/Quote]

请问“向下转换”要怎么理解?能具体说明下吗?谢谢。
blueliuyun 2011-08-10
  • 打赏
  • 举报
回复
[Quote=引用 3 楼 openxmpp 的回复:]
assistant上写了
类型兼容的话 尝试转换才会成功 否则返回0
QTimer是QtCore的方法,抽象按钮是UI的方法,一个可重入,一个不可重入,不一样的
[/Quote]
哦,O(∩_∩)O谢谢啊!
Qt的事件过滤器 Qt事件模型一个真正强大的特色是一个QObject 的实例能够管理另一个QObject 实例的事件。 让我们试着设想已经有了一个CustomerInfoDialog的小部件。CustomerInfoDialog 包含一系列QLineEdit. 现在,我们想用空格键来代替Tab,使焦点在这些QLineEdit间切换。 一个解决的方法是子类化QLineEdit,重新实现keyPressEvent(),并在keyPressEvent()里调用focusNextChild()。像下面这样: void MyLineEdit::keyPressEvent(QKeyEvent *event) { if (event->key() == Qt::Key_Space) { focusNextChild(); } else { QLineEdit::keyPressEvent(event); } } 但这有一个缺点。如果CustomerInfoDialog里有很多不同的控件(比如QComboBox,QEdit,QSpinBox),我们就必须子类化这么多控件。这是一个烦琐的任务。 一个更好的解决办法是: 让CustomerInfoDialog去管理他的子部件的按键事件,实现要求的行为。我们可以使用事件过滤器。 一个事件过滤器的安装需要下面2个步骤: 1, 调用installEventFilter()注册需要管理的对象。 2,在eventFilter() 里处理需要管理的对象的事件。 一般,推荐在CustomerInfoDialog的构造函数中注册被管理的对象。像下面这样: CustomerInfoDialog::CustomerInfoDialog(QWidget *parent) : QDialog(parent){ ... firstNameEdit->installEventFilter(this); lastNameEdit->installEventFilter(this); cityEdit->installEventFilter(this); phoneNumberEdit->installEventFilter(this); } 一旦,事件管理器被注册,发送到firstNameEdit,lastNameEdit,cityEdit,phoneNumberEdit的事件将首先发送到eventFilter()。 下面是一个 eventFilter()函数的实现: bool CustomerInfoDialog::eventFilter(QObject *target, QEvent *event) { if (target == firstNameEdit || target == lastNameEdit || target == cityEdit || target == phoneNumberEdit) { if (event->type() == QEvent::KeyPress) { QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event); if (keyEvent->key() == Qt::Key_Space) { focusNextChild(); return true; } } } return QDialog::eventFilter(target, event); } 在上面的函数中,我们首先检查目标部件是否是 firstNameEdit,lastNameEdit,cityEdit,phoneNumberEdit。接着,我们判断事件是否是按键事件。如果事件是按键事件,我们把事件转换为QKeyEvent。接着,我们判断是否按下了空格键,如果是,我们调用focusNextChild(),把焦点传递给下一个控件。然后,返回,true通知Qt,我们已经处理了该事件。 如果返回false的话,Qt继续将该事件发送给目标控件,结果是一个空格被插入到QLineEdit中。 如果目标控件不是 QLineEdit,或者按键不是空格键,我们将把事件传递给基类的eventFilter()函数。 Qt提供5个级别

16,239

社区成员

发帖
与我相关
我的任务
社区描述
Qt 是一个跨平台应用程序框架。通过使用 Qt,您可以一次性开发应用程序和用户界面,然后将其部署到多个桌面和嵌入式操作系统,而无需重复编写源代码。
社区管理员
  • Qt
  • 亭台六七座
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧