Qt学习笔记-2 对象模型
标准 C++ 对象模型在运行时效率方面卓有成效,但是在某些特定问题域下的静态特性就显得捉襟见肘。GUI 界面需要同时具有运行时的效率以及更高级别的灵活性。为了解决这一问题,Qt “扩展”了标准 C++。所谓“扩展”,实际是在使用标准 C++ 编译器编译 Qt 源程序之前,Qt 先使用一个叫做 moc(Meta Object Compiler,元对象编译器)的工具,先对 Qt 源代码进行一次预处理(注意,这个预处理与标准 C++ 的预处理有所不同。Qt 的 moc 预处理发生在标准 C++ 预处理器工作之前,并且 Qt 的 moc 预处理不是递归的。),生成标准 C++ 源代码,然后再使用标准 C++ 编译器进行编译。如果你曾经为信号函数这样的语法感到奇怪(现在我们已经编译过一些 Qt 程序,你应当注意到了,信号函数是不需要编写实现代码的,那怎么可以通过标准 C++ 的编译呢?),这其实就是 moc 进行了处理之后的效果。
Qt 使用 moc,为标准 C++ 增加了一些特性:
- 信号槽机制,用于解决对象之间的通讯,这个我们已经了解过了,可以认为是 Qt 最明显的特性之一;
- 可查询,并且可设计的对象属性;
- 强大的事件机制以及事件过滤器;
- 基于上下文的字符串翻译机制(国际化),也就是 tr() 函数,我们简单地介绍过;
- 复杂的定时器实现,用于在事件驱动的 GUI 中嵌入能够精确控制的任务集成;
- 层次化的可查询的对象树,提供一种自然的方式管理对象关系。
- 智能指针(QPointer),在对象析构之后自动设为 0,防止野指针;
- 能够跨越库边界的动态转换机制。
- 通过
QObject
类,我们可以很方便地获得这些特征。。这些特征由moc帮助我们实现,而moc其实实现的是元对象系统( meta-object system)的机制。这是一个标准C+扩展,使得标准C+更适合于进行GUI编程。
在本节中,主要介绍Qt的对象树。QObject
是以对象树的形式组织起来的。当创建一个QObject
对象时,会看到QObject
的构造函数接收一个QObect
的指针作为参数,这个参数就是parent,也就是父对象指针。也就是说,在创建QObject
对象时,可以提供一个其父对象,我们创建的这个QObject
会被自动添加到父对象的children()
列表
QWidget
是能够在屏幕上显示的一切组件的父类。QWidget
继承自QObject
,因此也继承了这种对象树关系。一个孩子自动地成为父组件的一个子组件。因此,它会显示在父组件的坐标系统中,被父组件的边界剪裁。例如,当用户关闭一个对话框的时候,应用程序将其删除,那么,我们希望属于这个对话框的按钮、图标等应该一起被删除。事实就是如此,因为这些都是对话框的子组件。
当然,我们也可以自己删除子对象,它们会自动从其父对象列表中删除。
我们可以使用QObject::dumpObjectTree()
和QObject::dumpObjectInfo()
这两个函数进行这方面的调试。Qt引入对象树的概念,在一定程度上解决了内存问题。
当一个QObject
对象在对上创建时,Qt会同时为其创建一个对象树。不过,对象树中对象的顺序时没有定义的。意思就是,销毁这些对象的顺序是未定义的,但是,任何对象树中的Qobject
对象delect的时候,如果这个对象有parent,则自动将其从parent的children()
列表删除;如果有孩子,则自动delete每一个孩子。Qt保证没有QObject
会被delete两次,这是由析构顺序决定的。
如果QObject
在栈上创建,Qt保持同样的行为。正常情况习啊,这也不会发生什么问题。来看看下面的代码片段:
1 | { |
作为父组件的window和作为子组件的quit都是QObject
的子类(事实上,它们都是QWidget
的子类,而QWidget
是QObject
的子类)。这段代码是正确的,quit的析构函数不会被调用两次,因为标准C++要求,局部对象的析构顺序应该按照其创建顺序的相反过程。因此,这段代码在超出作用域时会先调用quit的析构函数,将其从父对象window的子对象列表中删除,然后才会再调用window的析构函数。
如果我们使用下面的代码:
1 | { |
情况就会不一样,析构顺序就有了问题。在上面代码中,作为父对象的window会首先被析构,因为它是最后一个创建的对象。在析构过程中,它会调用子对象列表中每一个对象的析构函数,也就是说,quit此时就被析构了。然后,代码继续执行,在 window 析构之后,quit 也会被析构,因为 quit 也是一个局部变量,在超出作用域的时候当然也需要析构。但是,这时候已经是第二次调用 quit 的析构函数了,C++ 不允许调用两次析构函数,因此,程序崩溃了。
由此我们看到,Qt 的对象树机制虽然帮助我们在一定程度上解决了内存问题,但是也引入了一些值得注意的事情。这些细节在今后的开发过程中很可能时不时跳出来烦扰一下,所以,我们最好从开始就养成良好习惯,在 Qt 中,尽量在构造的时候就指定 parent 对象,并且大胆在堆上创建。