Qt 之对象树与所有权

作者: 一去、二三里
个人微信号: iwaleon
微信公众号: 高效程序员

QObjects在一个对象树中组织他们自己。当创建一个QObject时,如果使用了其他对象作为其父对象,那么,它就会被添加到父对象的children()列表中。这样一来,当父对象被销毁时,这个QObject也会被销毁。事实表明,这个机制非常适合于管理GUI对象。例如:一个QShortcut(键盘快捷键)对象是相关窗口的一个子对象,所以,当用户关闭了这个窗口时,快捷键也会被销毁。

详细描述

QQuickItem是Qt Quick模块的基本视觉元素,从QObject继承。但有一个不同于QObject parent的visual parent的概念,一个item的visual parent不一定和它的object parent相同。可以参考助手中的Concepts - Visual Parent in Qt Quick获取更多信息。

QWidget是Qt Widgets模块的基本类,扩展了父-子关系。孩子通常成为一个部件,即:它会显示在其父窗体的坐标系统中,并且由其父窗体的边界灵活地进行剪裁。例如:当应用程序在关闭提示框(QMessageBox)之后要删除它时,消息框的按钮和标签也会被删除。这正是我们所期望的,因为按钮和标签是对话框的子部件。

当然,也可以自己删除子对象,他们会从父窗体中移除自己。例如:当用户移除一个工具栏时,它可能会导致应用程序删除QToolBar中的一个对象,在这种情况下,工具栏的QMainWindow父窗体将会检测变化,并重新配置其相应的屏幕空间。

当一个应用程序看上去或执行起来比较奇怪时,调试函数QObject:dumpObjectTree()和QObject::dumpObjectInfo()通常比较有用。

QObjects的构造/销毁顺序

当QObjects被创建在堆上(例如:使用new创建)时,他们可以用任何顺序构造一个树。之后,树中的对象可以用任何顺序被销毁。当树中的任何QObject被删除时,如果它有父对象,析构函数会自动从其父对象中把它删除。如果它有子对象,析构函数会自动删除它的每一个子对象。无论销毁的顺序是什么,QObject都不会被删除两次,

当QObject被创建在栈上时,同样的行为适用。通常情况下,销毁顺序仍然不会出现问题。看下面的代码:

int main()
{
    QWidget window;
    QPushButton quit("Quit", &window);
    ...
}

父窗口(window)和子部件(quit)都属于QObjects,因为QPushButton继承自QWidget,而QWidget继承自QObject。这段代码是正确的:quit的析构函数不会被调用两次,因为C++语言标准(ISO/IEC 14882:2003)指定本地对象的析构函数以和构造函数相反的顺序被调用。所以,子部件quit的析构函数首先被调用,在window调用析构之前,它已经从其父窗口window中删除自身。

如果我们交换顺序,如下所示:

int main()
{
    QPushButton quit("Quit");
    QWidget window;

    quit.setParent(&window);
    ...
}

在这种情况下,销毁的顺序会引起一个问题。父窗体的析构函数首先被调用,因为它最后被创建。然后调用子部件(quit)的析构函数,这是不正确的,因为quit是一个局部变量,当出了作用域后,其析构函数再次被调用,因此会出错。

  • 14
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 10
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

一去丶二三里

有收获,再打赏!

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

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

打赏作者

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

抵扣说明:

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

余额充值