当前位置: 首页 > news >正文

Qt 对象树详解:从原理到运用

1. 什么是对象树?

对象树是一种基于父子关系的对象管理机制。在 Qt 中,所有继承自 QObject 的类都可以参与到对象树中。

当一个对象被设置为另一个对象的父对象时,子对象会被添加到父对象的内部列表中,形成一种树状结构。

Qt 提供了一个调试工具方法 dumpObjectTree() ,可以帮助开发者打印对象树的结构。

    QObject* parent = new QObject();QObject* child = new QObject();child->setParent(parent);parent->dumpObjectTree();

1.1 对象树结构

在 Qt 中,对象树的每个节点是一个 QObject 对象,而 “边” 代表父子关系。具体来说:

  • 根节点:没有父对象的对象(即 parent == nullptr)。
  • 子节点:有父对象的对象,它们会被添加到父对象的 子对象列表 中。
  • 叶子节点:没有子对象的对象。

grandparent 是根节点,parent 是 grandparent 的子节点,child 是 parent 的子节点,形成了一个三层的树形结构。

1.2 为什么选择 new 而不是栈上分配?

如果对象是在栈上创建的(如 局部变量),当作用域结束时,该对象会自动销毁。可能导致以下问题:

  • 如果子对象比父对象先销毁,父对象的 子对象列表 可能会指向无效内存。直观表现可能为:子对象区域显示为空白状态。

通过 new 创建对象,可以确保对象的生命周期由对象树管理,而不是由作用域控制。 这样可以避免上述问题。

2. 对象树的工作原理

在 Qt 中,继承自 QObject 的对象能够参与到对象树的前提是 该类必须正确地声明 Q_OBJECT

对象树通过以下两个关键成员变量实现:

  • QObject* parent :指向父对象的指针;
  • QList<QObject*> children:存储子对象的列表。

每个 QObject 对象可以有一个父对象 parent,和有多个子对象 children;子对象会被自动添加到父对象的 children 列表中。

2.1 对象创建时

当创建一个新的 QObject 对象并指定父对象时,会发生:

  1. 新对象的 parent 指针被设置为指定的父对象;
  2. 新对象被自动添加到父对象的 children 列表中。
2.2 对象销毁时

当一个对象被销毁时,以下操作会发生:

  1. 该对象的所有子对象会被递归销毁;
  2. 该对象从其父对象的 children 列表中移除。

​ Qt 的实现确保了在子对象的析构函数中,它会调用父对象的相关方法(如 QObjectPrivate::removeChild()),将自己从父对象的 children 列表中移除;显式移除子对象,是为了避免父对象的 children 列表中残留无效指针,从而确保对象树的完整性。

class MyObject : public QObject
{Q_OBJECT
public:QString name; // 对象名称explicit MyObject(QString name, QObject *parent) : name(name), QObject(parent) {}~MyObject(){qDebug() << name << " entering destructor.";if (parent()) {qDebug() << name << " is being removed from parent's children list.";}}
};void CreateObjects()
{MyObject* grandparent = new MyObject(QString("grandparent"));MyObject* parent = new MyObject(QString("parent"), grandparent);MyObject* child = new MyObject(QString("child"), parent);delete grandparent;
}

实际销毁顺序是:child -> parent -> grandparent —— 根据打印信息,child 开始析构时 parent 仍然存在。

3. 对象树的其它作用

在 Qt 中,对象树的核心作用是 简化对象的生命周期管理。通过对象树,开发者无需过多担心内存泄露或悬空指针等问题。

“简化对象的生命周期管理” 在 “工作原理” 部分已经解释过,除此之外,对象树的作用还包括:

3.1 层次化组织对象

​ 在 GUI 应用程序中,对象之间的关系通常是层次化的。例如,一个窗口可能包含多个控件,而每个控件又可能包含子控件。对象树提供了一种自然的方式来组织这些对象之间的关系。

int main(int argc, char *argv[])
{QApplication a(argc, argv);Widget w;w.show();return a.exec();
}
  1. Widget 是一个继承自 QWidget 的类,而 QWidget 又继承自 QObject。因此 Widget 对象可以参与到 Qt 的对象树机制中。

  2. 代码中 w 是一个在栈上分配的对象。

    如果用户手动关闭窗口(通过点击窗口的关闭按钮 或 调用 w.close()),窗口会被隐藏(调用了 hide() 方法),但 w 本身不会被销毁,因此它的子对象也不会被销毁 —— 栈上分配的对象由 C++ 的作用域管理,只有当作用域结束时(即 main() 函数返回时),w 才会被销毁。

  3. 设置 Qt::WA_DeleteOnClose

    Qt::WA_DeleteOnClose 是一个窗口属性,用于指示窗口在关闭时自动删除自身。如果设置了该属性,当用户手动关闭窗口时,w 会被销毁,并触发对象树的递归销毁。

    Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
    {ui->setupUi(this);// 设置关闭时自动销毁this->setAttribute(Qt::WA_DeleteOnClose);
    }
    
3.2 其它作用
  • 支持信号与槽机制:

    Qt 的信号与槽机制依赖于对象树来确保对象的生命周期一致性。

    如果一个对象被销毁,Qt 会自动断开与其相关的信号和槽连接;

    这一机制由 QObject 的析构函数触发,确保信号不会发送到已销毁的对象。

  • 简化资源管理:

    通过将资源封装成 QObject 的子类,并将其添加到对象树中,可以确保资源在父对象销毁时被正确释放。

    class FileWrapper : public QObject
    {Q_OBJECT
    public:FileWrapper(const QString& path, QObject* parent = nullptr): QObject(parent), file(path){}~FileWrapper(){file.close();}
    private:QFile file;
    };QObject* parent = new QObject();
    FileWrapper* file = new FileWrapper("test.txt", parent);
    
  • 提供遍历和查找功能:通过 QObject::children() 方法,可以获取某个对象的所有子类对象。

  • 避免重复释放:对象树通过集中管理对象的销毁过程,避免了重复释放的问题。

4. 注意事项
  • 避免循环引用:如果两个对象互相设置对方为父对象,会导致循环引用,从而引发内存泄露。

    解决方案: 确保父子关系是单向的。

  • 避免跨线程操作:如果父子对象位于不同的线程中,销毁顺序可能会受到线程调度的影响.

    QThread thread;
    QObject* obj = new QObject();
    obj->moveToThread(&thread);
    QObject* child = new QObject(obj);
    

    如果子对象在父对象销毁后仍然被访问(如 调用 child->parent() ),会导致未定义行为。

    解决方案:

    确保父子对象始终位于同一个线程;

    如果需要跨线程操作,可以使用信号与槽机制进行线程间通信,而不是直接操作对象树。

  • 避免手动干预销毁:如果手动调用 delete 销毁一个对象,而该对象同时参与对象树,可能会导致重复释放的问题。

    QObject* parent = new QObject();
    QObject* child = new QObject(parent);delete parent; // 自动销毁 child
    delete child;  // 再次销毁 child,重复释放
    

    解决方案:

    让对象树完全管理对象的生命周期,避免手动调用 delete ;

    如果必须手动管理对象,确保不将其添加到对象树中。

相关文章:

Qt 对象树详解:从原理到运用

1. 什么是对象树&#xff1f; 对象树是一种基于父子关系的对象管理机制。在 Qt 中&#xff0c;所有继承自 QObject 的类都可以参与到对象树中。 当一个对象被设置为另一个对象的父对象时&#xff0c;子对象会被添加到父对象的内部列表中&#xff0c;形成一种树状结构。 Qt 提…...

【软路由】ImmortalWrt 编译指南:从入门到精通

对于喜欢折腾路由器&#xff0c;追求极致性能和定制化的玩家来说&#xff0c;OpenWrt 无疑是一个理想的选择。而在众多 OpenWrt 衍生版本中&#xff0c;ImmortalWrt 以其更活跃的社区、更激进的特性更新和对新硬件的支持而备受关注。 本文将带你深入了解 ImmortalWrt&#xff0…...

【智能音频新风尚】智能音频眼镜+FPC,打造极致听觉享受!【新立电子】

智能音频眼镜&#xff0c;作为一款将时尚元素与前沿科技精妙融合的智能设备&#xff0c;这种将音频技术与眼镜形态完美结合的可穿戴设备&#xff0c;不仅解放了用户的双手&#xff0c;更为人们提供了一种全新的音频交互体验。新立电子FPC在智能音频眼镜中的应用&#xff0c;为音…...

第2章 windows故障排除(网络安全防御实战--蓝军武器库)

网络安全防御实战--蓝军武器库是2020年出版的&#xff0c;已经过去3年时间了&#xff0c;最近利用闲暇时间&#xff0c;抓紧吸收&#xff0c;总的来说&#xff0c;第2章开始带你入门了&#xff0c;这里给出了几个windows重要的工具&#xff0c;说实话&#xff0c;好多我也是第一…...

深度学习笔记——线性回归的从0开始实现

记录学习到的知识&#xff1a; 语义分割是将标签或类别与图片的每个像素关联的一种深度学习算法。 它用来识别构成可区分类别的像素集合。 图像分割是一个端到端图像分析过程&#xff0c;它将数字图像分成多个片段&#xff0c;并对每个区域中包含的信息进行分类。三种图像分割…...

配置Spring Boot中的Jackson序列化

配置Spring Boot中的Jackson序列化 在开发基于Spring Boot的应用程序时&#xff0c;Jackson是默认的JSON序列化和反序列化工具。它提供了强大的功能&#xff0c;可以灵活地处理JSON数据。然而&#xff0c;Jackson的默认行为可能无法完全满足我们的需求。例如&#xff0c;日期格…...

AWS跨账号服务全解析:安全共享资源的最佳实践

在复杂的云环境中,企业常常需要将不同业务部门、项目或环境分配到独立的AWS账户中,以实现资源隔离和权限管控。然而,跨账户的资源共享与协作需求也随之而来。AWS为此提供了丰富的跨账号服务,允许不同账户之间安全、高效地共享资源。本文将深入解析这些服务,并结合实际场景…...

Rust~String、str、str、String、Box<str> 或 Box<str>

Rust语言圣经中定义 str Rust 语言类型大致分为两种&#xff1a;基本类型和标准库类型&#xff0c;前者由语言特性直接提供&#xff0c;后者在标准库中定义 str 是唯一定义在 Rust 语言特性中的字符串&#xff0c;但也是几乎不会用到的字符串类型 str 字符串是 DST 动态大小…...

SpringBoot五:JSR303校验

精心整理了最新的面试资料和简历模板&#xff0c;有需要的可以自行获取 点击前往百度网盘获取 点击前往夸克网盘获取 松散绑定 意思是比如在yaml中写的是last-name&#xff0c;这个和lastName意思是一样的&#xff0c;-后的字母默认是大写的 JSR303校验 就是可以在字段增加…...

Oracle 数据库基础入门(四):分组与联表查询的深度探索(上)

在 Oracle 数据库的学习进程中&#xff0c;分组查询与联表查询是进阶阶段的重要知识点&#xff0c;它们如同数据库操作的魔法棒&#xff0c;能够从复杂的数据中挖掘出有价值的信息。对于 Java 全栈开发者而言&#xff0c;掌握这些技能不仅有助于高效地处理数据库数据&#xff0…...

基于SpringBoot的绿城郑州爱心公益网站设计与实现现(源码+SQL脚本+LW+部署讲解等)

专注于大学生项目实战开发,讲解,毕业答疑辅导&#xff0c;欢迎高校老师/同行前辈交流合作✌。 技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、小程序、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;…...

创建一个简单的spring boot+vue前后端分离项目

一、环境准备 此次实验需要的环境&#xff1a; jdk、maven、nvm和node.js 开发工具&#xff1a;idea或者Spring Tool Suite 4&#xff0c;前端可使用HBuilder X&#xff0c;数据库Mysql 下面提供maven安装与配置步骤和nvm安装与配置步骤&#xff1a; 1、maven安装与配置 1…...

标签使用笔记

文章目录 文件夹结构可以有多个功能吗?标签是如何保存的 标签做成对外接口保存、修改查询删除标签列表标签表设计标签和分类的区别 虽然大体知道怎么设计做&#xff0c;但是整理出来更清晰&#xff0c;那么整理下。 一般来说有两种索引就够。 1、标题文字索引。 # 用于搜索文章…...

Unity图集使用事项

一. 图集布局算法 紧密填充是一种常见的图集布局算法&#xff0c;它的主要目标是尽可能地减少图集的空间浪费。该算法会根据图像的形状和大小&#xff0c;将它们紧密地排列在图集中&#xff0c;以确保最小化空白区域的存在。这样可以有效地利用内存&#xff0c;并减少图集的尺…...

Flutter 学习之旅 之 flutter 在 Android 端读取相册图片显示

Flutter 学习之旅 之 flutter 在 Android 端读取相册图片显示 目录 Flutter 学习之旅 之 flutter 在 Android 端读取相册图片显示 一、简单介绍 二、简单介绍 image_picker 三、安装 image_picker 四、简单案例实现 五、关键代码 代码说明&#xff1a; 一、简单介绍 Fl…...

RagFlow专题二、RagFlow 核心架构(数据检索、语义搜索与知识融合)

深入解析 RagFlow 核心架构:数据检索、语义搜索与知识融合 在前一篇文章中,我们对 RagFlow 的核心理念、与传统 RAG 的区别以及其适用场景进行了深入探讨。我们了解到,RagFlow 通过动态优化检索、增强生成质量以及实时知识管理,使得大模型在复杂任务中的表现更加稳定和高效…...

解决各大浏览器中http地址无权限调用麦克风摄像头问题(包括谷歌,Edge,360,火狐)后续会陆续补充

项目场景&#xff1a; 在各大浏览器中http地址调用电脑麦克风摄像头会没有权限&#xff0c;http协议无法使用多媒体设备 原因分析&#xff1a; 为了用户的隐私安全&#xff0c;http协议无法使用多媒体设备。因为像摄像头和麦克风属于可能涉及重大隐私问题的API&#xff0c;ge…...

【SpringBoot+Vue】博客项目开发二:用户登录注册模块

后端用户模块开发 制定参数交互约束 当前&#xff0c;我们使用MybatisX工具快速生成的代码中&#xff0c;包含了一个实体类&#xff0c;这个类中包含我们数据表中的所有字段。 但因为有些字段&#xff0c;是不应该返回到前端的&#xff0c;比如用户密码&#xff0c;或者前端传…...

(十 二)趣学设计模式 之 享元模式!

目录 一、 啥是享元模式&#xff1f;二、 为什么要用享元模式&#xff1f;三、 享元模式的实现方式四、 享元模式的优缺点五、 享元模式的应用场景六、 总结 &#x1f31f;我的其他文章也讲解的比较有趣&#x1f601;&#xff0c;如果喜欢博主的讲解方式&#xff0c;可以多多支…...

leetcode第77题组合

原题出于leetcode第77题https://leetcode.cn/problems/combinations/ 1.树型结构 2.回溯三部曲 递归函数的参数和返回值 确定终止条件 单层递归逻辑 3.代码 二维数组result 一维数组path void backtracking(n,k,startindex){if(path.sizek){result.append(path);return ;}…...

Windows Cleaner终极攻略:系统优化与空间释放完整指南

Windows Cleaner终极攻略&#xff1a;系统优化与空间释放完整指南 【免费下载链接】WindowsCleaner Windows Cleaner——专治C盘爆红及各种不服&#xff01; 项目地址: https://gitcode.com/gh_mirrors/wi/WindowsCleaner Windows Cleaner是一款专为Windows系统设计的开…...

intv_ai_mk11应用场景:研发团队用其自动生成Git Commit Message规范模板

研发团队如何用intv_ai_mk11自动生成Git Commit Message规范模板 1. 研发团队的Commit Message痛点 每个研发团队都面临过这样的困境&#xff1a;代码提交信息五花八门&#xff0c;格式混乱。有的同事写"修复bug"&#xff0c;有的写"改了东西"&#xff0…...

AtCoder Beginner Contest 429

【赛时五题】AtCoder Beginner Contest 429 https://www.bilibili.com/video/BV1gXsZz8ELL/ 【赛时6题】AtCoder Beginner Contest 429 https://www.bilibili.com/video/BV1gXsZz8EZQ/ Atcoder Beginner Contest 429 https://www.bilibili.com/video/BV1SosZzdENX/ https://blo…...

SSM+Vue医院食堂订餐系统源码+论文

代码可以查看文章末尾⬇️联系方式获取&#xff0c;记得注明来意哦~&#x1f339; 分享万套开题报告任务书答辩PPT模板 作者完整代码目录供你选择&#xff1a; 《SpringBoot网站项目》1800套 《SSM网站项目》1500套 《小程序项目》1600套 《APP项目》1500套 《Python网站项目》…...

告别黑屏和错位!Uniapp视频轮播最佳实践:巧用v-if与swiper事件实现无缝切换

Uniapp视频轮播组件深度优化&#xff1a;从黑屏错位到无缝体验的全链路解决方案 在移动应用开发中&#xff0c;视频轮播组件已经成为提升用户参与度的关键元素。然而&#xff0c;当Uniapp开发者尝试在swiper组件中嵌入视频时&#xff0c;常常会遇到视频位置偏移、黑屏闪现、自动…...

砸钱做AI却看不见回报?实测实在Agent,上千位全球高管给出的标准答案

作为深耕B2B企服与AI产品评测领域的“老兵”&#xff0c;我在企服AI产品测评局的一线实操中见过太多令人唏嘘的案例。时间来到2026年4月1日&#xff0c;站在这个节点回望&#xff0c;过去一年全球企业在生成式AI上的投入堪称疯狂——仅美国企业在2025年的花费就预计高达370亿美…...

vue3 diff算法中的-双端 Diff + 最长递增子序列 讲解

一句话总结 Vue3 Diff 双端比较&#xff08;快速复用&#xff09; 最长递增子序列&#xff08;最小移动 DOM&#xff09; 目的&#xff1a;在乱序节点中&#xff0c;只移动最少 DOM&#xff0c;实现最高效更新。1. 先搞懂&#xff1a;Vue3 对比 Vue2 差在哪&#xff1f; Vue2…...

DOL-CHS-MODS整合包零基础精通指南:从安装到定制全方位教程

DOL-CHS-MODS整合包零基础精通指南&#xff1a;从安装到定制全方位教程 【免费下载链接】DOL-CHS-MODS Degrees of Lewdity 整合 项目地址: https://gitcode.com/gh_mirrors/do/DOL-CHS-MODS 项目价值定位 DOL-CHS-MODS作为Degrees of Lewdity的中文整合方案&#xff0…...

C++笔记 继承关系中构造和析构顺序(面向对象)

在C面向对象编程中&#xff0c;继承是实现代码复用和类层次设计的核心特性。当存在基类与派生类的继承关系时&#xff0c;构造函数和析构函数的调用顺序有严格的规则——这不仅是面试高频考点&#xff0c;更是避免内存泄漏、保证对象正确初始化/清理的关键。核心结论先明确&…...

三星 Infinite AI 葡萄酒冰箱:智能厨房新尝试能否突围?

AI 加持&#xff0c;葡萄酒管理新体验周一&#xff0c;三星推出了 Infinite AI 葡萄酒冰箱&#xff0c;目前仅在韩国有售。这款冰箱采用了“AI 葡萄酒管理器”&#xff0c;借助安装在顶部的“AI 视觉”摄像头&#xff0c;能检测用户放入或取出的酒瓶及位置&#xff0c;还能分析…...