Qt提供了对象树机制,能够自动、有效的组织和管理继承自QObject的Qt对象。
每个继承自QObject类的对象通过它的对象链表(QObjectList)来管理子类对象,当用户创建一个子对象时,其对象链表相应更新子类对象信息,对象链表可通过children()获取。
当父对象析构的时候,其对象链表中的所有(子)对象也会被析构,父对象会自动将其从父对象列表中删除。Qt 保证没有对象会被 delete 两次。开发中手动回收资源时建议使用deleteLater代替delete,因deleteLater多次是安全的,而delete多次是不安全的。
示例
新建QWidget项目。添加四个类,分别继承QLable、QPushButton、QRadioButton、QGridLayout
class MyLabel : public QLabel;
class MyLayout : public QGridLayout;
class MyPushButton : public QPushButton;
class MyRadioButton : public QRadioButton;
每个子类声明构造和析构函数,函数实现中仅使用qDebug( )输出标识句
//MyLabel类
MyLabel::MyLabel(QWidget *parent):QLabel(parent)
{
qDebug()<<"MyLabel构造"<<this;
}
MyLabel::~MyLabel()
{
qDebug()<<"MyLabel析构"<<this;
}
//MyLayout类
MyLayout::MyLayout(QWidget *parent):QGridLayout(parent)
{
qDebug()<<"MyLayout构造"<<this;
}
MyLayout::~MyLayout()
{
qDebug()<<"MyLayout析构"<<this;
}
//MyPushButton类
MyPushButton::MyPushButton(QWidget *parent):QPushButton(parent)
{
qDebug()<<"MyPushButton构造"<<this;
}
MyPushButton::~MyPushButton()
{
qDebug()<<"MyPushButton析构"<<this;
}
//MyRadioButton类
MyRadioButton::MyRadioButton(QWidget *parent):QRadioButton(parent)
{
qDebug()<<"MyRadioButton构造"<<this;
}
MyRadioButton::~MyRadioButton()
{
qDebug()<<"MyRadioButton析构"<<this;
}
在main函数中,初始化控件和布局管理器,将当前窗体设为控件和布局管理器的父窗体;将控件添加到布局管理器中
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
Widget w;
//当前窗体设为父窗体
MyLabel *myLabel =new MyLabel(&w);
MyPushButton*myBtn=new MyPushButton(&w);
MyRadioButton*myRbtn=new MyRadioButton(&w);
MyLayout*myLayout=new MyLayout(&w);
//设置文本
myLabel->setText("子标签");
myBtn->setText("子按钮");
myRbtn->setText("子单选");
//控件添加到布局管理器中
myLayout->addWidget(myLabel,0,0);
myLayout->addWidget(myBtn,1,0);
myLayout->addWidget(myRbtn,2,0);
return a.exec();
}
运行结果:
我们可以看到:在初始化时,当前窗体会先执行构造,随后是其子类执行构造;当窗体关闭时,当前窗体会先执行析构,随后是其子类执行析构,也就是说程序会自动、有效的组织和管理继承自QObject的Qt对象。 只要父类是QObject下的派生类,当父类被销毁或者创建时,其子类也会跟着创建和销毁。
我们可以使用.children( )函数查看一个类的派生类。这里我们查看当前窗体的派生类。
const QObjectList listW=w.children();
qDebug()<<"w.children()";
foreach (QObject* obj, listW) {
qDebug()<<obj;
}
运行结果:
从运行结果我们可以看到: 当我们对控件和布局管理器初始化时,已经将当前窗体设为其父窗体。这些控件和布局管理器也就顺应成章成为当前窗体的子类。
我们继续对标签控件派生子类
//创建一个sun布局管理器,将标签控件设为其父窗体
MyLayout*sunLayout=new MyLayout(myLabel);
//创建三个按钮控件,这里并未指明其父窗体
MyPushButton*sunBtn1=new MyPushButton;
MyPushButton*sunBtn2=new MyPushButton;
MyPushButton*sunBtn3=new MyPushButton;
//设置文本
sunBtn1->setText("孙按钮1");
sunBtn2->setText("孙按钮2");
sunBtn3->setText("孙按钮3");
//将按钮添加到布局管理器中
sunLayout->addWidget(sunBtn1);
sunLayout->addWidget(sunBtn2);
sunLayout->addWidget(sunBtn3);
//按钮初始时未设定父类,只是将其放入sun布局管理器中。
//sun布局管理器的父窗体设定为myLabel,当myLabel指定其布局管理器为孙布局管理器时,按钮会重新将myLabel设定为其父窗体
myLabel->setLayout(sunLayout);
继续使用.children( )查看标签类的派生类
//查看w子类
const QObjectList listW=w.children();
qDebug()<<"w.children()";
foreach (QObject* obj, listW) {
qDebug()<<obj;
}
//查看myLabel子类
const QObjectList listLabel=myLabel->children();
qDebug()<<"myLabel.children()";
foreach (QObject* obj, listLabel) {
qDebug()<<obj;
}
执行结果: