【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.mzph.cn/news/888123.shtml

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

相关文章

Vue进阶之单组件开发与组件通信

书接上篇&#xff0c;我们了解了如何快速创建一个脚手架&#xff0c;现在我们来学习如何基于vite创建属于自己的脚手架。在创建一个新的组件时&#xff0c;要在新建文件夹中打开终端创建一个基本的脚手架&#xff0c;可在脚手架中原有的文件中修改或在相应路径重新创建&#xf…

深度学习的python基础(1)

一.tensor创建 1.张量的定义 张量在形式上就是多维数组&#xff0c;例如标量就是0维张量&#xff0c;向量就是一维张量&#xff0c;矩阵就是二维张量&#xff0c;而三维张量就可以想象RGB图片&#xff0c;每个channel是一个二维的矩阵&#xff0c;共有三个channel&#xff0…

scss文件内引入其他scss文件报错

在 Sass (SCSS) 中&#xff0c;import 语句用于在当前文件中导入其他 Sass 文件&#xff0c;以便你可以重用样式和变量等。然而&#xff0c;从 Dart Sass 1.23.0 版本开始&#xff0c;import 语句已经被标记为弃用&#xff08;deprecated&#xff09;&#xff0c;并计划在未来的…

Unity3D模型场景等测量长度和角度功能demo开发

最近项目用到多段连续测量物体长度和角度功能&#xff0c;自己研究了下。 1.其中向量角度计算&#xff1a; 需要传入三个坐标来进行计算。三个坐标确定两条向量线段的方向&#xff0c;从而来计算夹角。 public Vector3 SetAngle(Vector3 p1, Vector3 p2,Vector3 p3) { …

【MATLAB】基于RSSI的蓝牙定位与例程,设置4个基站、二维定位

目录 ​编辑 商品描述 主要功能 技术细节 适用场景 下载链接 商品描述 这款基于接收信号强度指示&#xff08;RSSI&#xff09;原理的蓝牙定位程序&#xff0c;专为需要高效、可靠定位解决方案的开发者和研究人员设计。它能够在二维平面内&#xff0c;通过4个锚点实现对未…

【学习笔记】基于RTOS的设计中的堆栈溢出(Stack Overflow)-第1部分

本文由RTOS专家Jean J. Labrosse撰写。 基于RTOS的应用程序中的每个任务都需要自己的堆栈,堆栈的大小取决于任务的要求(例如,函数调用嵌套、传递给函数的参数、局部变量等)。 为了避免堆栈溢出,开发人员需要过度分配堆栈空间,但不要太多,以避免浪费RAM。 什么是堆栈溢…

ROC曲线

文章目录 前言一、ROC的应用&#xff1f;二、使用方式1. 数据准备2.绘图可视化 前言 在差异分析中&#xff0c;ROC曲线可以用来评估不同组之间的分类性能差异。差异分析旨在比较不同组之间的特征差异&#xff0c;例如在基因表达研究中比较不同基因在不同条件或组织中的表达水平…

基数排序(代码+注释)

#include <stdio.h> #include <stdlib.h>// 获取数组中的最大值 int GetMax(int* a, int n) {int max a[0];for (int i 1; i < n; i) {if (a[i] > max) {max a[i];}}return max; }// 对数组按照某个位数进行计数排序 void CountingSortForRadix(int* a, i…

第一届“吾杯”网络安全技能大赛 Writeup

战队信息 战队名称&#xff1a;在你眼中我是誰&#xff0c;你想我代替誰&#xff1f; 战队排名&#xff1a;13 Misc Sign Hex 转 Str&#xff0c;即可得到flag。 原神启动&#xff01; 不好评价&#xff0c;stegsolve 秒了&#xff1a; WuCup{7c16e21c-31c2-439e-a814-b…

AJAX一、axios使用,url组成(协议,域名,资源路径)查询参数和化简,错误处理,请求/响应报文,状态码,接口文档,

一、AJAX是什么 概念 &#xff1a; AJAX是一种与服务器&#xff08;后端&#xff09;通信的技术 二、请求库axios的基本用法 1导包 2使用 // 1. 发请求 axios({ url: 请求地址 }).then(res > { // 2.接收并使用数据 }) <body><p class"province"…

基于 Python、OpenCV 和 PyQt5 的人脸识别上课打卡系统

大家好&#xff0c;我是Java徐师兄&#xff0c;今天为大家带来的是基于 Python、OpenCV 和 PyQt5 的人脸识别上课签到系统。该系统采用 Python 语言开发&#xff0c;开发过程中采用了OpenCV框架&#xff0c;Sqlite db 作为数据库&#xff0c;系统功能完善 &#xff0c;实用性强…

智能化业务校验框架:动态设计与应用实践

目录 一、业务背景 二、配置内容展示 三、商品动态配置内容展示 &#xff08;一&#xff09;商品spu校验信息数据 &#xff08;二&#xff09;商品sku校验信息数据 &#xff08;三&#xff09;组包商品校验信息数据 &#xff08;四&#xff09;商品数据校验数据持有者 &…

idea2024加载flowable6.8.1.36遇到的问题-idea启动flowable问题flowable源码启动问题

代码下载地址&#xff1a; https://gitee.com/hanpenghu_admin_admin/flowable6.8.1.git 1.首先是通过顶层目录maven clean install 发现很多子模块并不会install本地mavenStore库&#xff0c;这导致了&#xff0c;一堆相互依赖的模块报错找不到&#xff0c;所以需要根据报错…

web安全从0到1:burp-suite3

声明&#xff01; 学习视频来自B站up主 **泷羽sec** 有兴趣的师傅可以关注一下&#xff0c;如涉及侵权马上删除文章&#xff0c;笔记只是方便各位师傅的学习和探讨&#xff0c;文章所提到的网站以及内容&#xff0c;只做学习交流&#xff0c;其他均与本人以及泷羽sec团队无关&a…

深度学习模型: BERT(Bidirectional Encoder Representations from Transformers)详解

一、引言 自然语言处理&#xff08;NLP&#xff09;领域在过去几十年取得了显著的进展。从早期基于规则的方法到统计机器学习方法&#xff0c;再到如今基于深度学习的模型&#xff0c;NLP 不断向着更高的准确性和效率迈进。BERT 的出现为 NLP 带来了新的突破&#xff0c;它能够…

ESP8266 (ESP-01S)烧录固件 和 了解与单片机通信必需的AT指令

ESP8266&#xff08;ESP-01s&#xff09;烧录固件 工具&#xff1a; 需要安装的原装出厂固件库&#xff1a; ESP8266 --接线-- VCC 3.3&#xff08;外接开发板&#xff09; GND GND&#xff08;外接开发板&#xff09; IO0 GND&#xff08;外接…

<三>51单片机PWM开发SG90和超声测距

目录 1,PWM开发SG90 1.1简介 1.2控制舵机 1.3编程实现 2,超声测距 2.1简介 2.2,超声波测距代码实现 1,PWM开发SG90 1.1简介 PWM&#xff0c;英文名Pulse Width Modulation&#xff0c;是脉冲宽度调制缩写&#xff0c;它是通过对一系列脉冲的宽度进 行调制&#xff0c;等…

【Python爬虫五十个小案例】爬取猫眼电影Top100

博客主页&#xff1a;小馒头学python 本文专栏: Python爬虫五十个小案例 专栏简介&#xff1a;分享五十个Python爬虫小案例 &#x1f40d;引言 猫眼电影是国内知名的电影票务与资讯平台&#xff0c;其中Top100榜单是影迷和电影产业观察者关注的重点。通过爬取猫眼电影Top10…

springboot信息化在线教学平台的设计与实现(代码+数据库+LW)

摘要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了信息化在线教学平台的开发全过程。通过分析信息化在线教学平台管理的不足&#xff0c;创建了一个计算机管理信息化在线教学平台的方案。文章介绍了信息化在线教…

Hadoop批量计算实验

参考: Hadoop(一)之实验一CentOS7配置Hadoop系统:配置CentOS和下载安装包_基于虚拟机cents7搭建hadoop实验目的-CSDN博客 --------------------------------------------------------- 一、安装Vmware 二、创建虚拟机 1.安装centos7 ①打开VMware,点击新建虚拟机。 …