一文详解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;
}

执行结果:

以上就是一文详解Qt中的对象树机制的详细内容,更多关于Qt对象树机制的资料请关注编程宝库其它相关文章!

 一、set/map的底层结构1、set/map的源码扒一扒STL库中set和map的底层结构,不难发现,set和map的底层用的都是红黑树且均为key/value模型。只不过set的key/ ...