Qt 智能指针
Qt 智能指针
文章目录
- Qt 智能指针
- `QScopedPointer`
- 1. 自动删除对象
- 2. 转移所有权
- 3. 管理私有数据
- `QSharedPointer`
- 关键特性
- 注意事项
- `QWeakPointer`
- 注意事项
- `QPointer`
QScopedPointer
QScopedPointer
是 Qt 提供的一个智能指针,主要用于简化资源管理,防止内存泄漏和悬挂指针问题。它属于 Qt 的内存管理工具,能够自动处理对象的生命周期,确保对象在超出作用域时被销毁。QScopedPointer
是基于 C++11 标准中的 std::unique_ptr
实现的,但它具有 Qt 的特点,通常用于局部对象的管理。
- 自动删除对象:当
QScopedPointer
超出作用域时,它会自动释放所持有的对象。这意味着无需手动delete
对象。 - 不能复制:
QScopedPointer
不支持复制操作,防止发生意外的多个指针指向同一个对象的问题。 - 所有权转移:可以使用
reset()
或通过构造函数将QScopedPointer
的所有权转移给另一个QScopedPointer
。
1. 自动删除对象
QScopedPointer
最常见的用法是在函数或局部作用域内管理动态分配的对象。在作用域结束时,QScopedPointer
自动销毁对象,无需显式调用 delete
。
#include <QScopedPointer>
#include <QDebug>class MyClass {
public:MyClass() { qDebug() << "MyClass constructed"; }~MyClass() { qDebug() << "MyClass destructed"; }
};void testScopedPointer() {QScopedPointer<MyClass> ptr(new MyClass);// 当函数返回时,ptr 超出作用域,对象会被自动销毁
} // 在这里,MyClass 对象会被自动删除
2. 转移所有权
QScopedPointer
不支持复制操作,但可以通过 reset()
或构造函数转移所有权。这样,QScopedPointer
可以在不同的作用域之间传递资源。
#include <QScopedPointer>
#include <QDebug>class MyClass {
public:MyClass() { qDebug() << "MyClass constructed"; }~MyClass() { qDebug() << "MyClass destructed"; }
};void transferOwnership() {QScopedPointer<MyClass> ptr1(new MyClass);// 将所有权从 ptr1 转移到 ptr2QScopedPointer<MyClass> ptr2(ptr1.take());// 现在 ptr1 不再拥有 MyClass 对象,ptr2 拥有它// ptr1 不再指向对象,但对象仍然存在,由 ptr2 管理
} // 在这里,ptr2 超出作用域时,MyClass 对象会被自动删除
3. 管理私有数据
在 Qt 的许多类中,私有数据(通常是一个包含实现细节的类)被封装在一个 QScopedPointer
中。这样可以确保私有数据在类的析构函数中自动释放,同时保持代码的简洁性和安全性。
示例:QFile
类
class QFilePrivate : public QIODevicePrivate {// 私有数据成员
};class QFile : public QIODevice {
public:QFile();~QFile();private:QScopedPointer<QFilePrivate> d_ptr;
};
在这个例子中,QFile
类使用 QScopedPointer
来管理 QFilePrivate
对象。当 QFile
对象析构时,QScopedPointer
会自动删除 QFilePrivate
对象,确保内存被释放。
QSharedPointer
QSharedPointer
是通过引用计数来管理对象的生命周期的,多个 QSharedPointer
对象可以共享同一个资源。每当 QSharedPointer
的拷贝构造或赋值操作发生时,引用计数会增加,而当一个 QSharedPointer
被销毁时,引用计数会减少。当引用计数降到 0 时,所指向的对象会自动被删除。
#include <QSharedPointer>
#include <QDebug>class MyClass {
public:void print() { qDebug() << "Hello from MyClass!"; }
};int main() {// 创建 QSharedPointer 对象,管理 MyClass 对象的生命周期QSharedPointer<MyClass> ptr1(new MyClass);// 创建另外一个 QSharedPointer,并共享 ptr1 所管理的对象QSharedPointer<MyClass> ptr2 = ptr1;// 使用 ptr1 和 ptr2 都能访问同一个对象ptr1->print();ptr2->print();// 不需要手动释放内存,当最后一个 QSharedPointer 被销毁时,MyClass 对象会自动删除return 0;
}
关键特性
- 引用计数:
QSharedPointer
通过引用计数来管理对象的生命周期。每当有新的QSharedPointer
对象指向相同的资源时,引用计数会增加;当某个QSharedPointer
对象销毁时,引用计数会减少。 - 自动销毁:当最后一个引用计数为 1 的
QSharedPointer
被销毁时,指向的对象会被自动删除,从而避免了内存泄漏。 - 线程安全:
QSharedPointer
的引用计数操作是线程安全的,但它本身并不保证被指向的对象本身是线程安全的。如果多个线程访问同一个QSharedPointer
对象,必须确保其他线程同步访问该对象。
注意事项
QSharedPointer
的引用计数机制在某些情况下可能导致循环引用问题,特别是当两个或更多的对象相互持有对方的QSharedPointer
时。此时,即使这些对象不再使用,引用计数也不会降到零,因为它们互相引用,导致对象无法被销毁,从而产生内存泄漏。- 解决方法:使用
QWeakPointer
来打破循环引用。QWeakPointer
是一种弱引用,持有一个QSharedPointer
对象,但它不会增加引用计数。当QSharedPointer
被销毁时,QWeakPointer
自动变为空指针。
- 解决方法:使用
- 不要混用裸指针和
QSharedPointer``QSharedPointer
需要确保它是唯一的内存管理者。如果你在程序中同时使用裸指针和QSharedPointer
管理相同的内存,可能会导致双重释放或内存泄漏。因此,避免裸指针与智能指针共享同一资源,确保对象始终由智能指针管理。
QWeakPointer
QWeakPointer
是 QSharedPointer
的一种补充,它本身不拥有对象的所有权。QWeakPointer
仅在 QSharedPointer
的引用计数为非零时提供访问该对象的能力,但不会阻止对象的销毁。换句话说,QWeakPointer
允许你引用一个对象而不会使得该对象无法销毁。
QWeakPointer
的主要特点:
- 弱引用:
QWeakPointer
不增加对象的引用计数,也就是说它不会阻止对象的销毁。 - 防止循环引用:
QWeakPointer
解决了QSharedPointer
可能导致的循环引用问题。 - 安全的访问方式:
QWeakPointer
可以通过toStrongRef()
方法转换为QSharedPointer
,从而安全地访问目标对象。
QWeakPointer
和 QSharedPointer
的配合
QWeakPointer
通常与 QSharedPointer
一起使用,用于避免循环引用。在有些情况下,两个对象会互相引用,导致它们的引用计数始终不为零,进而导致内存泄漏。QWeakPointer
可以打破这个循环引用链,它允许对象 A 持有对象 B 的 QWeakPointer
,而对象 B 可以持有对象 A 的 QSharedPointer
,从而确保对象 A 和 B 的生命周期由 QSharedPointer
管理。
QWeakPointer
的常见用法
下面是一个使用 QWeakPointer
的具体示例:
class B; // Forward declarationclass A {
public:QSharedPointer<B> b; // B的共享指针
};class B {
public:QWeakPointer<A> a; // A的弱引用
};int main() {QSharedPointer<A> a(new A); // 创建A对象QSharedPointer<B> b(new B); // 创建B对象a->b = b; // A持有B的共享指针b->a = a; // B持有A的弱引用return 0; // 程序退出时,A和B会被自动销毁,避免内存泄漏
}
注意事项
使用QWeakPointer时候,一定要使用isNULL
判断一下 资源是否释放
QSharedPointer<MyClass> shared(new MyClass(20));
QWeakPointer<MyClass> weak(shared);qDebug() << "Shared pointer value:" << shared->getValue();
qDebug() << "Weak pointer value:" << weak.data()->getValue();shared.clear(); // 删除 shared 指向的对象
// 此时,MyClass 对象的引用计数为 0,将被自动删除,而此时 QWeakPointer 对象 weak 也为 null。if (weak.isNull()) { // 判断 weak 是否为 nullqDebug() << "Weak pointer is null - object has been deleted"; // 执行
}
else {qDebug() << "Weak pointer is not null - object still exists";
}
QPointer
QPointer
是一个用于指向 Qt 对象(例如 QObject
的子类)的模板类,它会自动管理对象的生命周期。当一个 QObject
被销毁时,QPointer
会将其指针设为 nullptr
,这使得程序能够检测到所指向的对象已经被删除,从而避免访问已删除的对象,避免悬空指针问题。QPointer
只能用来管理 QObject
或其子类的对象。如果你需要管理其他类型的对象,可以考虑使用其他智能指针,如 std::shared_ptr
或 std::unique_ptr
。
#include <QPointer>
#include <QPushButton>
#include <QVBoxLayout>
#include <QWidget>
#include <QDebug>int main(int argc, char *argv[]) {QCoreApplication a(argc, argv);QWidget window;QVBoxLayout *layout = new QVBoxLayout(&window);QPushButton *button = new QPushButton("Click me");QPointer<QPushButton> pButton(button);layout->addWidget(button);window.show();QObject::connect(button, &QPushButton::clicked, [&] {if (pButton) {qDebug() << "Button exists, text:" << pButton->text();} else {qDebug() << "Button has been deleted";}});// 模拟按钮删除QObject::connect(button, &QPushButton::clicked, [&] {delete button;});return a.exec();
}
相关文章:
Qt 智能指针
Qt 智能指针 文章目录 Qt 智能指针QScopedPointer1. 自动删除对象2. 转移所有权3. 管理私有数据 QSharedPointer关键特性注意事项 QWeakPointer注意事项 QPointer QScopedPointer QScopedPointer 是 Qt 提供的一个智能指针,主要用于简化资源管理,防止内…...
CODESYS MODBUS TCP通信(禾川Q1 PLC作为MODBUS TCP从站)
禾川Q1 PLC MODBUS TCP 通信(PLC作为MODBUS TCP通信主站) 禾川Q1 PLC MODBUS TCP通信(CODESYS平台完整配置+代码)-CSDN博客文章浏览阅读28次。MATLAB和S7-1200PLC水箱液位高度PID控制联合仿真(MODBUSTCP通信)_将matlab仿真导入plc-CSDN博客文章浏览阅读722次。本文详细介绍了如…...

10.STM32F407ZGT6-内部温度传感器
参考: 1.正点原子 前言: 本笔记的主要目的和意义就是,再次练习ADC的使用。 32.1 内部温度传感器简介 STM32F407 有一个内部的温度传感器,可以用来测量 CPU 及周围的温度(TA)。对于STM32F407 系列来说,该温度传感器在…...

运维安全中心(堡垒机)
阿里云运维安全中心(Alibaba Cloud Operation and Maintenance Security, OMS,通常也称为“堡垒机”)是一款针对云上运维管理的安全解决方案。它专注于加强云环境中运维过程的安全性,确保对关键云资源和服务的访问可控、可审计&am…...
Linux OOM | Early OOM | 进程监视
注: 本文为 “Linux OOM” 相关文章合辑。 Linux OOM 终结者 译者:花名有孚 | 2015-07-21 08:47 现在是早晨 6 点钟。已经醒来的我正在总结到底是什么事情使得我的起床闹铃提前了这么多。我们的监控系统显示,Plumbr 服务出故障了。 现在我…...

【2024年华为OD机试】(A卷,100分)- 等和子数组最小和(Java JS PythonC/C++)
一、问题描述 题目描述 给定一个数组nums,将元素分为若干个组,使得每组和相等,求出满足条件的所有分组中,组内元素和的最小值。 输入描述 第一行输入 m 接着输入m个数,表示此数组nums 数据范围:1<m&…...
NFS服务
nfs文件系统 NFS:NetworkFileSystem网络文件系统,基于内核的文件系统。 服务安装 不固定端口启动,会注册到rpcbind(固定端口)服务上, 局域网适用[rootvm ~]# yum -y install nfs-utils # 依赖安装rpcbind [ro…...
RabbitMQ 交换机、队列和路由键的命名规范
在 RabbitMQ 中,使用 Topic Exchange 模式时,交换机、队列和路由键的命名规范是非常重要的,尤其是在多环境和多微服务的场景中。合理的命名规范可以提高消息系统的可维护性、可扩展性以及可读性。以下是一些关于 Topic Exchange 模式中交换机…...

腾讯云AI代码助手编程挑战赛-刑说
作品简介 鉴于当代普法力度不够大,这个刑说可以帮助大家更好的普及法律知识 技术架构 采用了全后端分离的架构,前端使用Vue.js,腾讯云的AI服务处理自然语言理解与生成。 实现过程 开发环境、开发流程 系统:win11 开发工具&…...

【测试】持续集成CI/CD
近期更新完毕,建议关注收藏点赞~ 目录 概括gitJenkinspostman集成jenkins代码集成jenkins 概括 CI/CD stands for Continuous Integration and Continuous Deployment 定义 团队成果持续集成到公共平台。一天可以集成1次or多次 本地代码管理 git 远程代…...
阿里云直播Web
官方文档:Web播放器SDK常见问题_视频点播(VOD)-阿里云帮助中心 bug:播流的不稳定,直播总会进入 onM3u8Retry 监听,用户端就会黑屏,(但其实并没有关播,正常关播进入的是pause这个监听࿰…...

DuckDB:PRAGMA语句动态配置数据库行为
PRAGMA语句是DuckDB从SQLite中采用的SQL扩展。PRAGMA命令可能会改变数据库引擎的内部状态,并可能影响引擎的后续执行或行为。本文介绍PRAGMA命令及其典型应用场景。 DuckDB PRAGMA介绍 在 DuckDB 中,PRAGMA 是一种编译指示(compiler directi…...
GO通过SMTP协议发送邮件
什么是SMTP协议 SMTP(Simple Mail Transfer Protocol,简单邮件传输协议)是用于发送邮件的协议。当一个邮件服务器需要发送邮件给另一个邮件服务器时,它会使用SMTP协议与目标服务器建立连接,并传输邮件内容。SMTP协议的…...
轻量自高斯注意力机制LSGAttention模型详解及代码复现
模型背景 近年来,卷积神经网络(CNN)在高光谱图像分类领域取得了显著进展。然而,CNN面临 长距离关系建模 和 计算成本 增加的挑战。为解决这些问题,研究人员提出了基于 轻量自高斯注意(Light Self-Gaussian-Attention, LSGA) 机制的视觉转换器(Vision Transformer, VIT),旨…...
解读若依框架中的`@Excel` 和 `@Excels` 注解
文章目录 一、Excels 注解详解1.1 适用场景1.2 作用与好处 二、Excel 注解详解2.1 核心属性解析2.2 高级用法2.3 综合应用案例 三、总结 解读若依框架中的 Xss 注解博客:解读若依框架中的 Xss 注解 接下来我们将对若依框架中的 Excel 和 Excels 注解进行更加详细的…...

云商城--基础数据处理和分布式文件存储
第2章 基础数据处理和分布式文件存储 1.分布式文件存储系统Ceph学习 1).掌握Ceph架构 2).掌握Ceph组件 3).搭建Ceph集群(了解) 2.Ceph使用 1).基于Ceph实现文件上传 2).基于Ceph实现文件下载 3.SKU、SPU管理 1).掌握SKU和SPU关系 2).理解商品发…...
六十九:基于openssl实战验证RSA
RSA(Rivest-Shamir-Adleman)是一种非对称加密算法,广泛应用于数据加密和数字签名领域。在实际开发和学习过程中,理解 RSA 的工作原理和使用场景非常重要。本文将以 OpenSSL 工具为基础,通过实例操作来验证和理解 RSA 的…...
Three.js 用户交互:构建沉浸式3D体验的关键
文章目录 前言一、基本交互:鼠标与触摸事件二、高级交互:键盘控制与游戏手柄支持三、物理模拟与碰撞检测四、手势识别与多点触控五、增强现实(AR)与虚拟现实(VR)六、触觉反馈与震动效果七、语音控制八、眼球…...
Android车机DIY开发之学习篇(五)默认应用修改
Android车机DIY开发之学习篇(五)默认应用修改 android默认应用位置 sdk/packages/apps InitRC配置 应用安装的目录 /system/priv-app 该路径存放一些系统底层的应用,比如Setting,systemUI等。该目录中的app拥有较高的系统权限,而且如果要使…...
linux 设置mysql 外网访问
1、修改 MySQL 配置文件 找到并编辑配置文件:在Linux系统中,MySQL的配置文件通常是/etc/mysql/my.cnf,使用命令sudo vim /etc/mysql/my.cnf打开文件。 注释或修改 bindaddress:找到bindaddress 127.0.0.1,将其注释掉…...
java_网络服务相关_gateway_nacos_feign区别联系
1. spring-cloud-starter-gateway 作用:作为微服务架构的网关,统一入口,处理所有外部请求。 核心能力: 路由转发(基于路径、服务名等)过滤器(鉴权、限流、日志、Header 处理)支持负…...
React Native 开发环境搭建(全平台详解)
React Native 开发环境搭建(全平台详解) 在开始使用 React Native 开发移动应用之前,正确设置开发环境是至关重要的一步。本文将为你提供一份全面的指南,涵盖 macOS 和 Windows 平台的配置步骤,如何在 Android 和 iOS…...
FFmpeg 低延迟同屏方案
引言 在实时互动需求激增的当下,无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作,还是游戏直播的画面实时传输,低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架,凭借其灵活的编解码、数据…...

【HarmonyOS 5.0】DevEco Testing:鸿蒙应用质量保障的终极武器
——全方位测试解决方案与代码实战 一、工具定位与核心能力 DevEco Testing是HarmonyOS官方推出的一体化测试平台,覆盖应用全生命周期测试需求,主要提供五大核心能力: 测试类型检测目标关键指标功能体验基…...

基于Flask实现的医疗保险欺诈识别监测模型
基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施,由雇主和个人按一定比例缴纳保险费,建立社会医疗保险基金,支付雇员医疗费用的一种医疗保险制度, 它是促进社会文明和进步的…...
postgresql|数据库|只读用户的创建和删除(备忘)
CREATE USER read_only WITH PASSWORD 密码 -- 连接到xxx数据库 \c xxx -- 授予对xxx数据库的只读权限 GRANT CONNECT ON DATABASE xxx TO read_only; GRANT USAGE ON SCHEMA public TO read_only; GRANT SELECT ON ALL TABLES IN SCHEMA public TO read_only; GRANT EXECUTE O…...
Caliper 配置文件解析:config.yaml
Caliper 是一个区块链性能基准测试工具,用于评估不同区块链平台的性能。下面我将详细解释你提供的 fisco-bcos.json 文件结构,并说明它与 config.yaml 文件的关系。 fisco-bcos.json 文件解析 这个文件是针对 FISCO-BCOS 区块链网络的 Caliper 配置文件,主要包含以下几个部…...

【Oracle】分区表
个人主页:Guiat 归属专栏:Oracle 文章目录 1. 分区表基础概述1.1 分区表的概念与优势1.2 分区类型概览1.3 分区表的工作原理 2. 范围分区 (RANGE Partitioning)2.1 基础范围分区2.1.1 按日期范围分区2.1.2 按数值范围分区 2.2 间隔分区 (INTERVAL Partit…...

佰力博科技与您探讨热释电测量的几种方法
热释电的测量主要涉及热释电系数的测定,这是表征热释电材料性能的重要参数。热释电系数的测量方法主要包括静态法、动态法和积分电荷法。其中,积分电荷法最为常用,其原理是通过测量在电容器上积累的热释电电荷,从而确定热释电系数…...
Java毕业设计:WML信息查询与后端信息发布系统开发
JAVAWML信息查询与后端信息发布系统实现 一、系统概述 本系统基于Java和WML(无线标记语言)技术开发,实现了移动设备上的信息查询与后端信息发布功能。系统采用B/S架构,服务器端使用Java Servlet处理请求,数据库采用MySQL存储信息࿰…...