1. QObject(parent) 核心含义

1. QObject(parent) 核心含义

cpp 复制代码
UdpSender::UdpSender(QObject *parent, ...)
    : QObject(parent)  // 初始化列表:调用父类QObject的构造函数
{
    // 函数体
}

这句代码的本质是:在创建 UdpSender 对象时,先调用它的父类 QObject 的构造函数,并将 parent 指针传递给父类

拆解说明:

  • : 后面是「构造函数初始化列表」:C++ 规定,子类构造时必须先初始化父类成员,这里就是显式调用父类 QObject 的构造函数;
  • QObject(parent):调用 QObject 类的构造函数 QObject(QObject *parent = nullptr),把传入的 parent 指针传给父类;
  • 因为 UdpSender 继承自 QObject,所以必须通过这种方式初始化父类(如果不写,编译器会自动调用 QObject() 无参构造,即 parent=nullptr)。

2. parent 指针的核心作用(Qt 父子对象机制)

这是 Qt 最核心的内存管理机制,parent 指针的作用只有一个:建立「父子对象关系」,让父对象自动管理子对象的内存

举个具体例子(结合我们的代码):

cpp 复制代码
// 在MainWindow中创建UdpSender,把this(MainWindow对象)作为parent
UdpSender *sender = new UdpSender(this); 

此时:

  • sender(UdpSender 对象)是「子对象」,MainWindow 对象是「父对象」;
  • MainWindow 对象被销毁时(比如关闭窗口),会自动调用 sender->deleteLater(),销毁这个 UdpSender 对象;
  • 你不需要手动写 delete sender;,Qt 会自动帮你释放内存,避免内存泄漏。

再看我们代码中的 QUdpSocket 创建:

cpp 复制代码
m_udpSocket = new QUdpSocket(this);

这里把 this(UdpSender 对象)作为 QUdpSocketparent,意味着:

  • m_udpSocketUdpSender 的子对象;
  • UdpSender 被销毁时,会自动销毁 m_udpSocket,无需手动释放。

3. 为什么必须写 QObject(parent)

(1)不写的后果

如果我们把初始化列表的 QObject(parent) 删掉:

cpp 复制代码
// 错误写法
UdpSender::UdpSender(QObject *parent, ...)
{
    // 编译器会自动调用 QObject() 无参构造,等价于 QObject(nullptr)
}

此时 UdpSender 对象的 parent 会被默认设为 nullptr,导致:

  • 该对象脱离 Qt 的父子管理体系,变成「孤儿对象」;
  • 你必须手动 delete 这个对象,否则会造成内存泄漏;
  • 无法利用 Qt 的信号槽机制的一些特性(比如线程间信号传递)。

(2)写了的好处

  1. 自动内存管理:父对象销毁时,子对象自动销毁,杜绝内存泄漏;
  2. 信号槽线程安全:父子对象在同一线程时,信号槽无需手动指定连接方式;
  3. 对象树遍历 :可以通过 parent() 获取父对象,或 children() 获取所有子对象(比如调试时查看对象关系)。

4. 完整示例:父子对象的生命周期

cpp 复制代码
#include "UdpSender.h"
#include <QApplication>
#include <QMainWindow>

int main(int argc, char *argv[]) {
    QApplication a(argc, argv);

    QMainWindow *window = new QMainWindow(); // 父对象
    UdpSender *sender = new UdpSender(window); // sender的parent是window

    window->show();
    int ret = a.exec();

    delete window; // 销毁window
    // 此时sender会被window自动销毁,无需手动delete sender;

    return ret;
}
  • 执行 delete window; 时,window 会遍历所有子对象,调用它们的 deleteLater()
  • sender 被销毁时,又会遍历它的子对象(比如 m_udpSocket),销毁 QUdpSocket
  • 整个过程无需手动管理子对象内存,这是 Qt 比原生 C++ 更易用的核心原因。

5. 补充:parent 的默认值

在我们的构造函数声明中,parent 有默认值 nullptr

cpp 复制代码
explicit UdpSender(QObject *parent = nullptr, ...);

这意味着:

  • 如果你创建对象时不传入 parent,比如 UdpSender *sender = new UdpSender();,则 parent=nullptr,该对象无父对象;
  • 如果你传入 parent,则建立父子关系,享受自动内存管理。

总结

  1. QObject(parent)调用父类 QObject 的构造函数 ,将 parent 指针传递给父类,建立父子对象关系;
  2. parent 指针的核心作用是Qt 自动内存管理:父对象销毁时,子对象自动销毁,避免内存泄漏;
  3. 继承自 QObject 的类(如 UdpSender/UdpReceiver),构造函数必须通过初始化列表调用 QObject(parent),否则会失去 Qt 的内存管理特性。
相关推荐
大黄说说8 分钟前
Go并发双雄:WaitGroup与Channel的抉择与协作
java·服务器·数据库
喜欢打篮球的普通人15 分钟前
MLIR入门
数据库·mlir
jjjava2.016 分钟前
数据库事务:ACID特性与实战应用
java·开发语言·数据库
东北甜妹33 分钟前
Redis Cluster 集群
数据库
(Charon)40 分钟前
【kv存储】基于 C 的 KV 存储项目:主从单向同步是怎么实现的
数据库
Jul1en_43 分钟前
【Redis】String 类型命令、编码方式与应用场景
数据库·redis·缓存
lifallen1 小时前
一篇文章讲透 Flink State
大数据·数据库·python·flink
赵渝强老师1 小时前
【赵渝强老师】MySQL数据库的分库与分表
数据库·mysql
XDHCOM1 小时前
利用MSSQL解析优化数据库性能,提升效率,驱动业务创新与稳定发展
数据库·sqlserver
·云扬·1 小时前
MySQL分区实战指南:从原理到落地的完整攻略
数据库·mysql