【C/C++】内存管理详解:从new/delete到智能指针的全面解析

文章目录

    • 更多文章
    • C/C++中的传统内存管理方式
      • `new`和`delete`运算符
      • `malloc`和`free`函数
      • 传统内存管理的弊端
    • 智能指针的崛起
      • 智能指针的定义与作用
      • C++11引入的标准智能指针
    • 详解C++标准智能指针
      • `std::unique_ptr`
        • 特点
        • 使用方法
        • 适用场景
      • `std::shared_ptr`
        • 特点
        • 使用方法
        • 适用场景
      • `std::weak_ptr`
        • 特点
        • 使用方法
        • 适用场景
    • 智能指针与传统内存管理的比较
      • 内存安全性
      • 性能考量
      • 使用复杂度
    • 智能指针的最佳实践
      • 选择合适的智能指针类型
      • 避免循环引用
      • 与传统指针的混用
      • 避免不必要的拷贝
      • 使用`make_*`函数
    • 案例教程:使用智能指针管理内存
      • 传统方法实现
      • 使用`std::unique_ptr`重构
      • 使用`std::shared_ptr`的场景
    • 更多文章
  • 结语

更多文章

【OpenAI】获取OpenAI API Key的多种方式全攻略:从入门到精通,再到详解教程!!

【VScode】VSCode中的智能编程利器,全面揭秘ChatMoss & ChatGPT中文版

【体验最新的GPT系列模型!支持Open API调用、自定义助手、文件上传等强大功能,助您提升工作效率!点击链接体验:CodeMoss & ChatGPT-AI中文版】
在这里插入图片描述

C/C++中的传统内存管理方式

在深入探讨智能指针之前,我们首先需要了解C/C++中传统的内存管理方式,以便更好地理解智能指针的优势所在。

newdelete运算符

在C++中,newdelete是用于动态分配和释放内存的运算符。

// 使用new分配内存
int* ptr = new int;// 使用delete释放内存
delete ptr;

通过new,程序员可以在堆上动态分配内存,而通过delete则可以手动释放这部分内存。然而,这种手动管理方式容易导致内存泄漏或悬挂指针等问题。

mallocfree函数

在C语言中,mallocfree函数用于动态内存分配和释放。

// 使用malloc分配内存
int* ptr = (int*)malloc(sizeof(int));// 使用free释放内存
free(ptr);

malloc函数返回一个void*指针,需要通过类型转换来使用。与new/delete类似,mallocfree也需要开发者手动管理内存,同样面临内存泄漏和其他相关问题。

传统内存管理的弊端

虽然new/deletemalloc/free为程序员提供了灵活的内存管理方式,但其手动管理的特性也带来了诸多弊端:

  1. 内存泄漏:如果开发者忘记调用delete或者free释放内存,程序会出现内存泄漏,长时间运行后可能导致系统资源耗尽。
  2. 悬挂指针:在释放内存后,指针仍然指向原来的地址,如果再次访问可能导致未定义行为。
  3. 异常安全性差:在异常发生时,手动管理的内存释放往往难以保证,容易导致资源泄漏。
  4. 代码复杂度高:需要在多个地方进行内存分配和释放,增加了代码的复杂性和维护难度。

这些问题不仅影响程序的稳定性和性能,还增加了开发和调试的难度。因此,寻求一种更安全、更高效的内存管理方式成为现代C++开发的重要课题。

智能指针的崛起

为了应对传统内存管理方式的诸多问题,C++11标准引入了智能指针,作为一种RAII机制,旨在自动管理内存资源,减少内存泄漏和其他相关问题的发生。
在这里插入图片描述

智能指针的定义与作用

智能指针是一种封装了普通指针的类,通过自动管理内存的分配和释放,简化了内存管理的过程。它们利用独占或共享所有权的概念,确保在对象不再使用时,自动释放相关资源,从而提高代码的安全性和可维护性。

智能指针的主要作用包括:

  1. 自动内存管理:避免手动调用deletefree,减少内存泄漏的风险。
  2. 异常安全:在异常发生时,智能指针能够确保资源被正确释放。
  3. 所有权管理:通过不同类型的智能指针,管理资源的独占或共享所有权,提高代码的表达力和安全性。

C++11引入的标准智能指针

C++11标准引入了三种主要的标准智能指针,分别适用于不同的使用场景:

  1. std::unique_ptr:独占所有权,不能被复制,适用于资源的独占管理。
  2. std::shared_ptr:共享所有权,多个shared_ptr可以指向同一资源,通过引用计数管理资源的生命周期。
  3. std::weak_ptr:观察者指针,不增加引用计数,主要用于解决shared_ptr之间的循环引用问题。

这些智能指针的引入极大地简化了内存管理过程,提升了代码的安全性和可维护性。

详解C++标准智能指针

为了全面掌握智能指针的使用,以下将对C++11标准中的三种主要智能指针进行详细解析,包括其特点、使用方法及适用场景。

std::unique_ptr

特点
  • 独占所有权unique_ptr拥有其所指向资源的独占所有权,不能被复制,只能被移动。
  • 轻量级:相比shared_ptrunique_ptr更为轻量,适用于简单的资源管理场景。
  • 自动释放:当unique_ptr生命周期结束时,自动调用delete释放资源,避免内存泄漏。
使用方法
#include <memory>void uniquePtrExample() {// 创建一个unique_ptrstd::unique_ptr<int> ptr(new int(10));// 访问指针std::cout << "Value: " << *ptr << std::endl;// 转移所有权std::unique_ptr<int> ptr2 = std::move(ptr);if (!ptr) {std::cout << "ptr is now null." << std::endl;}
}
适用场景
  • 资源的独占管理:适用于资源不需要被多个对象共享的场景,如单一对象的内部资源。
  • RAII模式:在资源管理的RAII模式中,unique_ptr是首选工具。
  • 性能要求高的场景:由于unique_ptr无引用计数,适用于对性能有严格要求的场景。

【体验最新的GPT系列模型!支持Open API调用、自定义助手、文件上传等强大功能,助您提升工作效率!点击链接体验:CodeMoss & ChatGPT-AI中文版】
在这里插入图片描述

std::shared_ptr

特点
  • 共享所有权:多个shared_ptr可以共同指向同一资源,通过内部的引用计数机制管理资源的生命周期。
  • 引用计数机制:每一个shared_ptr的拷贝都会增加引用计数,销毁时会减少引用计数,当引用计数为零时,自动释放资源。
  • 灵活性高:适用于多个对象需要共同管理同一资源的场景。
使用方法
#include <memory>void sharedPtrExample() {// 创建一个shared_ptrstd::shared_ptr<int> ptr1 = std::make_shared<int>(20);{// 复制shared_ptrstd::shared_ptr<int> ptr2 = ptr1;std::cout << "Value: " << *ptr2 << ", Use count: " << ptr1.use_count() << std::endl;}// ptr2超出作用域,引用计数减少std::cout << "Use count after ptr2 is destroyed: " << ptr1.use_count() << std::endl;
}
适用场景
  • 资源需要被多个对象共享:如共享的数据缓冲区、共享的资源池等。
  • 复杂的数据结构:如图结构、循环引用的场景(需要结合weak_ptr使用)。
  • 需要控制资源的生命周期:当资源的生命周期需要被多个部分共同管理时。

std::weak_ptr

特点
  • 非拥有性观察者指针weak_ptr不拥有资源的所有权,不影响引用计数。
  • 防止循环引用:在shared_ptr间可能产生的循环引用场景中,通过weak_ptr打破循环,避免内存泄漏。
  • 访问资源需转换:要访问资源,需先将weak_ptr转换为shared_ptr,确保资源在访问期间不会被释放。
使用方法
#include <memory>struct Node {std::shared_ptr<Node> next;std::weak_ptr<Node> prev; // 使用weak_ptr避免循环引用
};void weakPtrExample() {auto first = std::make_shared<Node>();auto second = std::make_shared<Node>();first->next = second;second->prev = first; // weak_ptr,不增加引用计数
}
适用场景
  • 打破循环引用:在涉及双向引用的数据结构中,如双向链表、图等。
  • 缓存系统:实现缓存时,weak_ptr可用于观察但不拥有缓存对象。
  • 临时访问:在需要临时访问资源但不希望延长其生命周期时。

智能指针与传统内存管理的比较

在了解了传统内存管理方式和智能指针的基本概念后,接下来将具体比较二者在内存安全性、性能和使用复杂度等方面的区别,为开发者选择合适的内存管理策略提供参考。

【体验最新的GPT系列模型!支持Open API调用、自定义助手、文件上传等强大功能,助您提升工作效率!点击链接体验:CodeMoss & ChatGPT-AI中文版】
在这里插入图片描述

内存安全性

传统方法

  • 手动管理内存,容易出现内存泄漏、悬挂指针等问题。
  • 开发者需要严格遵循内存分配和释放的规范,增加了出错的可能性。

智能指针

  • 自动管理内存,减少了人为忘记释放内存的风险。
  • 通过RAII机制,在对象生命周期结束时自动释放资源,提高了内存安全性。
  • weak_ptr有效防止了shared_ptr的循环引用问题。

性能考量

传统方法

  • 无额外的性能开销,适用于对性能有极高要求的场景。
  • 但由于手动管理,错误释放内存可能导致不可预测的性能问题。

智能指针

  • unique_ptr几乎无额外开销,适用于性能敏感的场景。
  • shared_ptr由于引用计数机制,存在一定的性能开销,尤其是在高频率的拷贝和销毁操作中。
  • weak_ptr本身开销较低,但在转换为shared_ptr时需要一定的计算。

使用复杂度

传统方法

  • 灵活性高,但需要开发者手动管理,增加了代码的复杂性和出错概率。
  • 在复杂的应用场景中,维护手动管理的代码较为困难。

智能指针

  • 提供了更高层次的抽象,简化了内存管理的流程。
  • 学习曲线相对较低,但需要理解不同智能指针的适用场景和使用方法。
  • 提高了代码的可读性和可维护性,减少了内存管理相关的出错概率。

智能指针的最佳实践

为了充分发挥智能指针的优势,开发者需要遵循一些最佳实践,合理选择和使用智能指针,避免潜在的问题。

选择合适的智能指针类型

  • std::unique_ptr:当资源拥有权是独占的,且不需要共享时,优先使用unique_ptr。它轻量且效率高,适合大多数独占资源的管理场景。

    std::unique_ptr<MyClass> ptr = std::make_unique<MyClass>();
    
  • std::shared_ptr:当资源需要被多个对象共享时,使用shared_ptr。确保没有不必要的共享所有权,以避免引用计数的额外开销。

    std::shared_ptr<MyClass> ptr1 = std::make_shared<MyClass>();
    std::shared_ptr<MyClass> ptr2 = ptr1;
    
  • std::weak_ptr:在需要观察但不拥有资源的场景中使用,特别是在打破shared_ptr循环引用时。

    std::weak_ptr<MyClass> weakPtr = sharedPtr;
    

避免循环引用

在某些场景,如树形结构、双向链表等,shared_ptr可能导致循环引用,进而引发内存泄漏。此时,应结合weak_ptr使用,打破循环引用。

struct Parent;
struct Child;struct Parent {std::shared_ptr<Child> child;
};struct Child {std::weak_ptr<Parent> parent; // 使用weak_ptr避免循环引用
};

与传统指针的混用

尽可能避免混用智能指针与原始指针,尤其是在管理同一资源时。若确实需要使用原始指针,确保它们仅作为观察者存在,不参与所有权管理。

std::unique_ptr<MyClass> ptr = std::make_unique<MyClass>();
MyClass* rawPtr = ptr.get(); // 仅作为观察者使用

避免不必要的拷贝

对于shared_ptr,避免不必要的拷贝操作,尤其是在高频率的函数调用中,因为每一次拷贝都会增加和减少引用计数,带来性能开销。

// 不推荐
void function(std::shared_ptr<MyClass> ptr);// 推荐
void function(const std::shared_ptr<MyClass>& ptr);

使用make_*函数

优先使用std::make_uniquestd::make_shared等工厂函数创建智能指针,避免手动使用new,提高代码的安全性和可读性。

auto ptr = std::make_unique<MyClass>();
auto sptr = std::make_shared<MyClass>();

案例教程:使用智能指针管理内存

通过一个实际案例,展示传统内存管理方式与智能指针的应用差异,帮助读者直观理解智能指针的优势。

传统方法实现

假设我们需要实现一个简单的类Person,并在主函数中动态创建和管理Person对象。

#include <iostream>
#include <string>class Person {
public:Person(const std::string& name) : name_(name) {std::cout << "Person " << name_ << " created." << std::endl;}~Person() {std::cout << "Person " << name_ << " destroyed." << std::endl;}void greet() const {std::cout << "Hello, I am " << name_ << "." << std::endl;}private:std::string name_;
};int main() {// 动态分配Person对象Person* person = new Person("Alice");person->greet();// 忘记释放内存,导致内存泄漏// delete person;return 0;
}

问题

  • 如果忘记调用delete person;,会导致内存泄漏。
  • 在异常发生时,delete可能无法被调用,进一步加剧内存泄漏的问题。

使用std::unique_ptr重构

通过使用std::unique_ptr,自动管理Person对象的生命周期,避免内存泄漏。

#include <iostream>
#include <string>
#include <memory>class Person {
public:Person(const std::string& name) : name_(name) {std::cout << "Person " << name_ << " created." << std::endl;}~Person() {std::cout << "Person " << name_ << " destroyed." << std::endl;}void greet() const {std::cout << "Hello, I am " << name_ << "." << std::endl;}private:std::string name_;
};int main() {{// 使用unique_ptr管理Person对象std::unique_ptr<Person> person = std::make_unique<Person>("Bob");person->greet();} // person超出作用域,自动调用deletereturn 0;
}

优势

  • 自动释放内存,无需手动调用delete
  • 即使在异常发生时,unique_ptr也能确保资源被正确释放。

使用std::shared_ptr的场景

假设我们有多个对象需要共享同一个Person对象,使用std::shared_ptr可以方便地管理共享所有权。

#include <iostream>
#include <string>
#include <memory>class Person {
public:Person(const std::string& name) : name_(name) {std::cout << "Person " << name_ << " created." << std::endl;}~Person() {std::cout << "Person " << name_ << " destroyed." << std::endl;}void greet() const {std::cout << "Hello, I am " << name_ << "." << std::endl;}private:std::string name_;
};void greetPerson(std::shared_ptr<Person> person) {person->greet();
}int main() {// 使用shared_ptr管理Person对象std::shared_ptr<Person> person = std::make_shared<Person>("Charlie");greetPerson(person);std::cout << "Use count: " << person.use_count() << std::endl;return 0;
}

优势

  • 通过shared_ptr,多个函数或对象可以共享同一个Person对象,而无需担心内存泄漏。
  • 当所有shared_ptr实例被销毁时,资源自动释放。

更多文章

【OpenAI】获取OpenAI API Key的多种方式全攻略:从入门到精通,再到详解教程!!

【VScode】VSCode中的智能编程利器,全面揭秘ChatMoss & ChatGPT中文版

结语

掌握内存管理不仅是C/C++开发者的必备技能,更是提升编程能力的重要一步。通过理解传统方法与智能指针的优劣,并灵活运用智能指针的各类工具,开发者能够编写出更加健壮、高效的代码,轻松应对复杂的开发挑战。希望本篇文章能够为你在内存管理的道路上提供有力的指导,助你在C/C++编程的世界中游刃有余。

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

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

相关文章

通过 SSH 进行WordPress网站的高级服务器管理

我在管理hostease的服务器时&#xff0c;时常需要通过SSH登录服务器进行修改。而在网站管理中&#xff0c;SSH不仅是一个基础工具&#xff0c;更是高级用户用来精细化管理和优化服务器的重要工具。通过SSH&#xff0c;你可以深入监控服务器的性能、精细管理系统资源&#xff0c…

MFC 对话框中显示CScrollView实例

有时候我们需要在对话框中显示CScrollView效果的控件&#xff0c;类似于以下效果&#xff1a; 使用实例可参考&#xff1a;MFC对话框显示CScrollView例子_哔哩哔哩_bilibili 创建CScrollView中显示的子对话框与子类&#xff1a; 两个对话框对应的类&#xff1a; CScrollView继…

vue3 ajax获取json数组排序举例

使用axios获取接口数据 可以在代码中安装axios包&#xff0c;并写入到package.json文件&#xff1a; npm install axios -S接口调用代码举例如下&#xff1a; const fetchScore async () > {try {const res await axios.get(http://127.0.0.1:8000/score/${userInput.v…

详解登录MySQL时出现SSL connection error: unknown error number错误

目录 登录MySQL时出错SSL connection error: unknown error number 出错原因 使用MySQL自带的工具登录MySQL 登陆之后&#xff0c;使用如下命令进行查看 解决方法 找到MySQL8安装目录下的my.ini配置文件 记事本打开my.ini文件&#xff0c;然后按下图所示添加配置 此时再…

mini-spring源码分析

IOC模块 关键解释 beanFactory&#xff1a;beanFactory是一个hashMap, key为beanName, Value为 beanDefination beanDefination: BeanDefinitionRegistry&#xff0c;BeanDefinition注册表接口&#xff0c;定义注册BeanDefinition的方法 beanReference&#xff1a;增加Bean…

linux内核读写硬盘文件 kernel_writekernel_read

简介 在内核中读取硬盘文件&#xff0c;内核5.10测试了下&#xff0c;可以正常运行。 代码 1. 使用filp_open打开文件 2. 使用kernel_write和kernel_read读写文件 3. 使用filp_close关闭文件 #include <linux/module.h> #include <linux/init.h> #include &l…

记录一次 k8s 节点内存不足的排查过程

背景&#xff1a;前端服务一直报404&#xff0c;查看k8s日志&#xff0c;没发现报错&#xff0c;但是发现pods多次重启。 排查过程&#xff1a; 查看pods日志&#xff0c;发现日志进不去。 kubectrl logs -f -n weave pod-name --tail 100查看pod describe kubectl describ…

微软要求 Windows Insider 用户试用备受争议的召回功能

拥有搭载 Qualcomm Snapdragon 处理器的 Copilot PC 的 Windows Insider 计划参与者现在可以试用 Recall&#xff0c;这是一项臭名昭著的快照拍摄 AI 功能&#xff0c;在今年早些时候推出时受到了很多批评。 Windows 营销高级总监 Melissa Grant 上周表示&#xff1a;“我们听…

嵌入式Qt使用ffmpeg视频开发记录

在此记录一下Qt下视频应用开发的自学历程&#xff0c;可供初学者参考和避雷。 了解常用音频格式yuv420p、h264等了解QML&#xff0c;了解QVideoOutput类的使用&#xff0c;实现播放yuv420p流参考ffmpeg官方例程&#xff0c;调用解码器实现h264解码播放 不需要手动分帧。ffmpeg…

Web day02 Js Vue Ajax

目录 1.javascript: 1.js的引入方式&#xff1a; 2.js变量 & 数据类型 & 输出语句&#xff1a; 模板字符串&#xff1a; 3.函数 & 自定义对象&#xff1a; 4. json 字符串 & DOM操作&#xff1a; 5. js事件监听&#xff1a; 6.js的模块化导入或者导出&a…

Qt Graphics View 绘图实例

Qt Graphics View 绘图实例 这个实例程序实现如下功能&#xff1a; 可以创建矩形、椭圆、三角形、梯形、直线、文字等基本图形。每个图形项都可以被选择和移动。图形项或整个视图可以缩放和旋转。图形项重叠时&#xff0c;可以调整前置或后置。多个图形项可以组合&#xff0c;…

TCP socket api详解 续

文章目录 守护进程怎么做到&#xff1f;setsid返回值 dev/null字符文件 daemonTCP协议 退出的时候呢&#xff1f; 会话有很多后台任务&#xff0c;bash肯定会退&#xff0c;那后台会话怎么办呢&#xff1f; 理论上也要退的&#xff0c;但实际上关了bash&#xff0c;bash肯定要…

06_数据类型

数据类型 数据类型分类 JavaScript 语言的每一个值,都属于某一种数据类型。JavaScript 的数据类型,共有六种。(ES6 又新增了第七种 Symbol 类型的值和第八种 BigInt类型,当前课程暂不涉及) 据类型分类 原始类型(基础类型) var age = 20, var name = 尚学堂"; var le…

scrapy豆瓣爬虫增强-批量随机请求头

1.1 豆瓣爬虫增强,中间件随机请求头 1.2 清除原有的中间件,进行中间件测试 1.3 导入全新的中间件 1.4 运行爬虫,这个时候的请求头是固定的 1.5 强化对agent的输出,会舍弃输出cookie,使输出更明了 1.6 转移输出请求头位置 新增输出 造成这样问题的原因是Douban/Douban/settings…

非关系型数据库有哪些特点?

非关系型数据库&#xff08;NoSQL&#xff09;具有以下主要特点‌&#xff1a;‌1 ‌灵活的数据存储方式‌&#xff1a;非关系型数据库不采用传统的基于表格的数据存储方式&#xff0c;而是采用更加灵活的数据存储方式。它可以存储各种类型的数据&#xff0c;包括文本、图像、音…

智慧防汛平台在城市生命线安全建设中的应用

随着城市化进程的加快&#xff0c;城市基础设施的复杂性和互联性不断增强&#xff0c;城市生命线的安全管理面临前所未有的挑战。智慧防汛平台作为城市生命线安全建设的重要组成部分&#xff0c;通过现代信息技术提升城市防汛应急管理的智能化水平&#xff0c;保障城市安全。 …

【ChatGPT大模型开发调用】如何获得 OpenAl API Key?

如何获取 OpenAI API Key 获取 OpenAI API Key 主要有以下三种途径&#xff1a; OpenAI 官方平台 (推荐): 开发者用户可以直接在 OpenAI 官方网站 (platform.openai.com) 注册并申请 API Key。 通常&#xff0c;您可以在账户设置或开发者平台的相关页面找到申请入口。 Azure…

vue3 发送 axios 请求时没有接受到响应数据

<script setup> import Edit from ./components/Edit.vue import axios from axios import { onMounted,ref } from vue// TODO: 列表渲染 //装数据的列表 const list ref([]) const count ref(0) const getList async () > {//通过发送 /list 请求从后端拿到列表数…

衡山派D133EBS 开发环境安装及SDK编译烧写镜像烧录

1.创建新文件夹&#xff0c;用来存放SDK包&#xff08;其实本质就是路径要对就ok了&#xff09;&#xff0c;右键鼠标通过Open Git Bash here来打开git 输入命令 git clone --depth1 https://gitee.com/lcsc/luban-lite.git 来拉取&#xff0c;如下所示&#xff1a;&#xff0…

关于Vscode配置Unity环境时的一些报错问题(持续更新)

第一种报错&#xff1a; 下载net请求超时&#xff08;一般都会超时很正常的&#xff09; 实际时并不需要解决&#xff0c;它对你的项目毫无影响 第二种报错&#xff1a; .net版本不匹配 解决&#xff1a;&#xff08;由于造成问题不一样&#xff0c;所以建议都尝试一次&…