【C++】编程规范之内存规则

在高质量编程中,内存管理是一个至关重要的方面。主要有以下原则:

内存分配后需要检查是否成功:内存分配可能会失败,特别是在内存紧张的情况下。因此,在分配内存后,应该检查分配是否成功。

int* ptr = new int;
if (ptr == nullptr) {// 内存分配失败的处理逻辑
} else {// 内存分配成功*ptr = 10;
}

指针在申请前和释放后需要置空:为了避免悬挂指针(Dangling Pointer)的问题,申请内存前和释放后应该将指针置为空。

int* ptr = nullptr; // 初始化为空指针ptr = new int;
// 使用ptrdelete ptr;
ptr = nullptr; // 释放内存后将指针置为空

内存使用前需要初始化:在使用动态分配的内存前,应该确保对其进行初始化,以避免访问未初始化的内存导致的未定义行为。

int* ptr = new int;
*ptr = 10; // 初始化内存
// 使用ptrdelete ptr;

自定义类对象和STL对象禁止使用memset/memcpy等内存操作:对于自定义类对象和STL容器等动态分配的内存,应该避免使用memset、memcpy等内存操作函数,而应该使用类的构造函数、赋值运算符等方法来进行内存操作。

#include <vector>class MyClass {
public:MyClass(int val) : value(val) {}
private:int value;
};int main() {std::vector<MyClass> vec;vec.push_back(MyClass(10)); // 正确的内存操作方式// 错误的方式:memset(&vec[0], 0, sizeof(MyClass)); // 禁止使用memsetreturn 0;
}

内存申请和释放需要匹配:确保每次内存分配后都有对应的释放操作,避免内存泄漏。

int* ptr = new int;
// 使用ptrdelete ptr; // 内存释放与分配匹配

资源申请和释放需要匹配:除了内存之外,其他资源如文件、锁等也需要在申请后及时释放,确保资源的正确管理。

#include <fstream>int main() {std::ifstream file("example.txt");if (!file.is_open()) {// 文件打开失败的处理逻辑} else {// 文件打开成功// 使用文件file.close(); // 关闭文件,释放资源}return 0;
}

在复杂的内存管理场景中。可以使用goto语句到一个标签,通常称为“出口”或“清理”标签,是一种确保资源释放的常见方法。这种方式通常被称为“资源获取即初始化(Resource Acquisition Is Initialization,RAII)”的编程范式。
如果任何一个内存分配失败,goto exit;语句将会跳转到清理标签,执行资源释放的操作,确保内存管理的匹配性。这种方法可以减少重复的错误处理代码,提高代码的可读性和可维护性。但需要谨慎使用goto语句,以避免引入混乱和不易理解的代码结构。


#include <iostream>void complexFunction() {int* ptr1 = nullptr;int* ptr2 = nullptr;ptr1 = new int;if (ptr1 == nullptr) {std::cerr << "Memory allocation failed for ptr1" << std::endl;goto exit; // 内存分配失败,跳转到清理标签}ptr2 = new int;if (ptr2 == nullptr) {std::cerr << "Memory allocation failed for ptr2" << std::endl;delete ptr1; // 删除ptr1已分配的内存goto exit; // 内存分配失败,跳转到清理标签}// 使用ptr1和ptr2exit:delete ptr1; // 释放内存delete ptr2; // 释放内存
}int main() {complexFunction();return 0;
}

内存申请和释放原则为谁申请谁释放:内存的申请和释放应该由同一个模块或者同一个函数负责,以确保管理的一致性。

void process() {int* ptr = new int;// 使用ptrdelete ptr; // 释放内存的职责由process函数负责
}

注意内存越界:在操作数组和指针时,需要格外小心,确保不会发生数组越界或指针越界的情况,以避免潜在的安全问题。

int arr[5];
for (int i = 0; i < 5; ++i) {arr[i] = i;
}
// 错误的访问方式:arr[5] = 10; // 内存越界

良好的内存管理是高质量编程的基础之一。遵循以上内存管理规则可以有效地提高程序的健壮性和可靠性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/791254.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【ZZULIOJ】1030: 判断直角三角形(Java)

目录 题目描述 输入 输出 样例输入 Copy 样例输出 Copy code 题目描述 输入三个正整数&#xff0c;判断用这三个整数做边长是否能构成一个直角三角形。 输入 输入三个正整数。 输出 能否构成直角三角形。如能输出&#xff1a;yes.若不能&#xff0c;输出&#xff1a…

java操作mongodb详解

前言 一切操作都应该以官方文档为准&#xff0c;mongodb官网文档地址&#xff1a; https://www.mongodb.com/docs/ &#xff0c;网上关于java操作mongodb的文章偏少&#xff0c;而且有些乱。这篇文章是在项目中使用mongodb后的一些总结&#xff0c;希望能帮到大家。 1.创建mon…

(译) 理解 Elixir 中的宏 Macro, 第四部分:深入化

Elixir Macros 系列文章译文 [1] (译) Understanding Elixir Macros, Part 1 Basics[2] (译) Understanding Elixir Macros, Part 2 - Macro Theory[3] (译) Understanding Elixir Macros, Part 3 - Getting into the AST[4] (译) Understanding Elixir Macros, Part 4 - Divin…

如何开启MySQL的binlog日志

1.启用远程连接&#xff1a; 如果你想要允许远程主机连接到MySQL服务器&#xff0c;需要进行以下步骤&#xff1a; 确保MySQL服务器的防火墙允许远程连接的流量通过。在MySQL服务器上&#xff0c;编辑MySQL配置文件&#xff08;一般是my.cnf&#xff09;&#xff0c;找到bind-…

Go——函数

一. 函数定义 1.1 特点 无需声明原型支持不定变参支持多返回值支持命名返回参数支持匿名函数和闭包函数也是一种类型&#xff0c;一种函数可以赋值给变量不支持嵌套&#xff0c;一个包不能有两个名字一样的函数不支持重载不支持默认参数 1.2 函数声明 函数声明包含一个函数名&…

备战蓝桥杯---DP刷题2

1.树形DP&#xff1a; 即问那几个点在树的直径上&#xff0c;类似ROAD那题&#xff0c;我们先求一下每一个子树根的子树的最大值与次大值用d1,d2表示&#xff0c;直径就是d1d2的最大值&#xff0c;那么我们如何判断是否在最大路径上&#xff0c;其实就是看一下从某一点出发的所…

还得是抖音,字节推出竖屏视频理解数据集,入选CVPR2024

ChatGPT狂飙160天&#xff0c;世界已经不是之前的样子。 新建了免费的人工智能中文站https://ai.weoknow.com 新建了收费的人工智能中文站https://ai.hzytsoft.cn/ 更多资源欢迎关注 短视频在当下社交媒体逐渐成为主导的视频格式。传统视频处理技术和研究一般都专注于横屏视频…

58商铺全新UI试客试用平台网站php源码

探索未来商铺新纪元&#xff0c;58商铺全新UI试客试用平台网站PHP源码完整版震撼来袭&#xff01; 在这个数字化飞速发展的时代&#xff0c;58商铺一直致力于为商家和消费者打造更加便捷、高效的交易平台。今天&#xff0c;我们荣幸地推出全新UI试客试用平台网站PHP源码完整版…

计算模型 观察分析 杂记

计算模式 计算模式通常指的&#xff1a;用特定计算资源完成特定计算任务所采用的计算策略。计算资源主要指运算器和存储器&#xff0c;当然若其他设备影响因素较大的情况下也考虑控制器&#xff0c;输入输出设备&#xff1b;计算任务多种多样&#xff0c;可以是简单的加减乘除&…

动态规划基础

动态规划 1、动态规划的概念 简称DP,通过把原问题分解为相对简单的子问题的方式求解复杂问题的方法。常常适用于有重叠子问题和最优子结构性质的问题。 简单来说,就是给定一个问题,把它拆成一个个子问题,查到子问题可以直接解决。然后把子问题答案保存起来,以减少重复计算…

Flink SQL系列之:解析Debezium数据格式时间字段常用的函数

Flink SQL系列之:解析Debezium数据格式时间字段常用的函数 一、FROM_UNIXTIME二、DATE_FORMAT三、TO_DATE四、CAST五、TO_TIMESTAMP_LTZ六、CONVERT_TZ七、FROM_UNIXTIME八、TO_TIMESTAMP九、常见用法案例1.案例一2.案例二3.案例三4.案例四5.案例五

C/C++ 项目:分别用精密星历和广播星历计算卫星坐标

文章目录 Part.I IntroductionChap.I rinex.hChap.II gmain_body.h Part.II 使用方法扩展阅读 Part.I Introduction 本文将介绍一个小项目的使用方法&#xff0c;此项目可用精密星历和广播星历计算卫星位置&#xff0c;并将两者结果做差&#xff0c;输出至文件。 其实 『分别…

SWM341系列应用(上位机应用)

SWM341系列之上位机应用 1、分级图像和PNG、JPG的应用 现象&#xff1a;客户使用SWM34SVET6HMI_0.4.1版本上位机进行UI界面布局&#xff0c;反馈在模拟运行时&#xff08;PC端&#xff09;流畅&#xff0c;在Demo平台&#xff08;设备端&#xff09;运行卡顿。 分析及解决&…

【fastadmin】脚本模式下,日志钩子函数执行出现死循环,导致内存溢出奔溃

问题出现原因是想对项目中error级别的日志&#xff0c;接入钉钉告警&#xff0c;方便查看 于是使用钩子方法&#xff0c;日志写入完成后&#xff0c;自动调用自定义的告警方法中 1、在application/tags.php 中添加log_write_done > [app\\common\\behavior\\Common, ],2、在…

【THM】Nmap Post Port Scans(后端口扫描)-初级渗透测试

介绍 本房间是 Nmap 系列的最后一个(网络安全简介模块的一部分)。在这个房间中,我们重点关注端口扫描之后的步骤:特别是服务检测、操作系统检测、Nmap脚本引擎和保存扫描结果。 Nmap实时主机发现Nmap基本端口扫描Nmap高级端口扫描Nmap后端口扫描在本系列的第一个房间中,我…

ZJGSU 1858在数组中查找两个数之和等于输入的另一个数

描述 题目&#xff1a;输入一个已经按升序排序过的数组和一个数字&#xff0c; 在数组中查找两个数&#xff0c;使得它们的和正好是输入的那个数字。如果有多对数字的和等于输入的数字&#xff0c;输出任意一对即可。 例如输入数组1、2、4、7、11、15和数字15。由于41115&…

代码随想录第29天|491.递增子序列 46.全排列 47.全排列 II

目录&#xff1a; 491.递增子序列 46.全排列 47.全排列 II 491.递增子序列 491. 非递减子序列 - 力扣&#xff08;LeetCode&#xff09; 代码随想录 (programmercarl.com) 回溯算法精讲&#xff0c;树层去重与树枝去重 | LeetCode&#xff1a;491.递增子序列_哔哩哔哩_bili…

DTFT及其反变换的直观理解

对于离散时间傅里叶变换(DTFT)及其反变换的讲解&#xff0c;教材里通常会先给出DTFT正变换的公式&#xff0c;再举个DTFT的简单变换例子&#xff0c;推导一下DTFT的性质&#xff0c;然后给出DTFT反变换的公式&#xff0c;再证明一下正变换和反变化的对应关系。总的来说就是&…

Spring-IoC 基于xml管理

现大多使用注解方式&#xff0c;xml方式并不简洁&#xff0c;本文仅记录xml用作基础学习。 0、前提 首先在父项目的pom.xml中配置好依赖们。然后子模块也可以使用这些依赖。 在resource目录下创建Spring的xml文件&#xff0c;名称无要求&#xff0c;本文使用bean.xml。文件最…

黄锈水过滤器 卫生热水工业循环水色度水处理器厂家工作原理动画

​ 1&#xff1a;黄锈水处理器介绍 黄锈水处理器是一种专门用于处理“黄锈水”的设备&#xff0c;它采用机电一体化设计&#xff0c;安装方便&#xff0c;操作简单&#xff0c;且运行费用极低。这种处理器主要由数码射频发生器、射频换能器、活性过滤体三部分组成&#xff0c;…