一文详解Qt中的对象树机制

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;
}

执行结果:

作者:音视频开发老舅原文地址:https://blog.csdn.net/m0_60259116/article/details/129676358

%s 个评论

要回复文章请先登录注册