「C++ 内存管理篇 1」C++动态内存分配

目录

〇、C语言的动态内存分配方式

一、C++的动态内存分配方式

1. 什么是C++的动态内存分配?

2. 为什么需要C++的动态内存分配?

a. new的优势

b. new的不足

c. delete的优势

d. 总结

3. 怎么使用new和delete?

a. 对于内置类型

b. 对于自定义类型 

c. 为什么new不需要检查失败?

4. new和delete的实现原理

5. new[]和delete[]的实现原理

7. 重载operator new与operator delete(了解)

8. 定位new表达式(placement-new) (了解)

9. malloc/free和new/delete的区别


〇、C语言的动态内存分配方式

        关于C语言的动态内存分配方式,简单来讲就是使用四个库函数:malloc、calloc、 realloc、free对堆区的内存进行灵活的分配和回收。有兴趣的话可以看看这篇文章:  「C语言进阶1」动态内存分配


一、C++的动态内存分配方式

1. 什么是C++的动态内存分配?

        动态内存分配是指在程序运行时,系统根据需要动态地申请和释放内存空间。
        C++中的动态内存分配是通过new和delete操作符来实现的。
        所以C++的动态内存分配简单来讲就是通过new和delete操作符对堆区的内存进行动态内存管理。

---------------------------------------------------------------------------------------------------------------------------------

2. 为什么需要C++的动态内存分配?

        C语言动态内存分配方式在C++中可以继续使用,但有些地方使用起来比较麻烦,比如创建动态对象时:如果选择使用malloc来创建动态对象,那就只是在堆上开辟了空间,没有初始化对象,我们又要想方设法对这个动态对象进行初始化。

        有没有办法在创建动态对象的同时对其完成初始化操作呢?所以C++又提出了自己的动态内存管理方式:通过操作符new和delete对堆区的内存进行进行动态内存管理。

a. new的优势

  • 创建和初始化一体:
            开辟空间和初始化在同一语句内,可以在开辟空间后按需求同时完成初始化操作,不像malloc只是完成空间的开劈,需要另起一行来初始化。
  • 操作统一:
            内置类型和自定义类型使用new创建和初始化动态变量的方法没有区别。
  • 对自定义类型会自动调用构造函数:
            对自定义类型,malloc只会分配内存空间,不会调用对象的构造函数。而new会在分配内存后自动调用对象的构造函数进行初始化。

  • 自动计算需要的内存大小
            malloc函数需要指定要分配的内存大小(以字节为单位),而new操作符会根据所需变量的类型自动计算所需的内存大小。
  • 类型安全:
            new操作符会进行类型检查,并返回类型正确的指针。而malloc函数,返回的是void*指针,需要自己转换为正确的类型。
  • 分配内存失败会进行异常处理:
            new操作符在分配内存失败时会抛出std::bad_alloc异常,可以通过异常处理机制来处理内存分配失败的情况。而malloc函数在分配内存失败时会返回NULL,需要手动检查返回值并处理。

  • 内存对齐
            malloc函数返回的内存地址是任意的,可能不满足特定的对齐要求。而new操作符会返回已对齐的内存地址,以确保对象的成员变量按照正确的对齐方式存储。


        综上所述,new操作符在C++中提供了更安全、更简洁、更易于管理的内存分配方式,与C++的面向对象特性和智能指针等高级功能紧密结合,使得内存管理更加高效和可靠。

b. new的不足

        new不支持扩容,一旦你使用 new(或 new[])分配了一定大小的内存,这块内存的大小就是固定的,你不能直接改变它的大小。如果你需要更大的内存空间,你需要手动进行内存管理,包括释放旧的内存块并分配一个新的、更大的内存块。

c. delete的优势

        相比于free,delete在释放动态对象的空间前,还会调用析构函数清理对象。 

d. 总结

        综上所述,为了更简洁、安全、方便的开辟和释放动态对象,C++引入了操作符new和delete来进行动态内存管理,它们不仅能完成开辟和释放空间的操作,对于自定义类型还会自动调用构造和析构函数完成初始化和清理工作,同时它们也更简便、更安全。

---------------------------------------------------------------------------------------------------------------------------------

3. 怎么使用new和delete?

a. 对于内置类型

//使用new动态的申请内存创建int变量,不初始化
int* a1 = new int;
//使用new动态的申请内存创建int变量,使用括号初始化
int* a2 = new int(2);

//使用new动态的申请内存创建int数组,不初始化
int* b1 = new int[10];
//使用new动态的申请内存创建int数组,使用{}初始化(C++11后引入)
int* b2 = new int[10]{ 1, 2 ,3, 4 };

//delete是关键字直接用就可以
//使用delete释放单个的动态变量
delete a1;
delete a2;
//使用delete[]释放连续的动态数组
delete[] b1;
delete[] b2;

        注意:申请和释放单个元素的空间,使用new和delete操作符,申请和释放连续的空间,使用new[]和delete[],一定要匹配起来使用。因为newnew[]以及deletedelete[]在内部执行的操作是不同的,因此它们不能互换使用。例如,如果你使用new分配了一个数组,但使用delete而不是delete[]来释放它,那么只有数组的第一个元素会被正确释放,其余的元素将保持未释放的状态,从而导致内存泄漏。

b. 对于自定义类型 

对于自定义类型,如果不显示初始化,那么会调用其默认构造函数进行初始化。

class A {
private:
    int _a;
    int _b;
public:
    A(int a = 1, int b = 1)
        : _a(1), _b(0)
    {
        _a = a;
        _b = b;
    }
};


// 用new创建一个自定义类型对象不初始化,编译器会调用默认构造函数
A* p0 = new A;
// 用new创建一个自定义类型对象并显示初始化
A* p1 = new A(1, 2);

// 用delete销毁自定义类型对象
delete p0;
delete p1;   

C++11后,可以用以下四种方式在new时对类数组初始化:

// 创建一个自定义类型对象数组(不显示初始化,编译器调用默认构造函数初始化)
A* p2 = new A[2];

// 创建一个自定义类型对象数组并初始化(C++11后支持)
A* p3 = new A[2]{ 1, 2 }; // 每个数对应一个对象,初始化对应对象的第一个成员
A* p4 = new A[2]{ (1,2,3,4,5)};  // 每个()对应一个对象,用()中的最后一个数初始化对应对象的第一个成员
A* p5 = new A[2]{ {1,2}, {3,4} }; // 每个{}对应一个对象,按顺序初始化对象成员
A* p6 = new A[2]{ A(1, 2), A(3, 4)}; // 每个构造函数对应一个对象

delete[] p2;
delete[] p3;
delete[] p4;
delete[] p5;
delete[] p6;

c. 为什么new不需要检查失败?

 malloc和new对于开辟空间失败的处理方式不同:
        malloc是一个函数,失败返回NULL;new是一个操作符,仅在动态内存分配成功时返回一个指针,分配失败只会抛异常不会返回空指针(除非使用了std::nothrow参数)。使用try{}catch{}语句来捕获并处理异常。


4. new和delete的实现原理

        new和delete是用户进行动态内存申请和释放的操作符,new在底层是调用operator new全局函数来申请空间,delete在底层通过operator delete全局函数来释放空间。operator new 和operator delete并不是在重载new和delete运算符,而是系统提供的全局函数用来申请和释放空间。以下是对该语句进行反汇编的结果:


当然new也不仅仅只是调用operator new全局函数来申请空间,还会调用构造函数,申请空间失败时还会抛异常:


所以在c++中更推荐使用new来动态申请空间,一是能同时调用构造函数,二是出错时抛异常,这样符合C++的失败机制。

5. new[]和delete[]的实现原理

        operator new 实际也是通过malloc来申请空间,如果 malloc申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供该措施 就继续申请,否则就抛异常。operator delete 最终是通过free来释放空间的。


        new[ ]的底层是先调用operator new[ ],再调用 operator new,完成N个对象的空间申请,最后调用N次构造函数。
        而delete[ ]的底层是先调用N次析构函数,完成对N个对象的清理,然后调用operator delete[ ],再调用 operator delete来释放空间。


7. 重载operator new与operator delete(了解)

        一般情况下不需要对 operator new 和 operator delete进行重载,除非在申请和释放空间时候有某些特殊的需求。比如:在使用new和delete申请和释放空间时,打印一些日志信息,可以简单帮助用户来检测是否存在内存泄漏。或是改用内存池,而不直接从堆上申请空间。

        new一个类时,看有没有自己的专属operator new,有,优先使用专属operator new,否则使用默认的全局operator new。


当频繁调用new时,想提高效率,不再走默认operator new中的malloc,而是自己定制一个内存池。使用内存池的优势在于,一次性向堆申请一大块空间A,以后要开辟空间直接在 A中拿即可,省时省力,而每次使用malloc,都要先申请,然后在堆中找一块合适的空间。


8. 定位new表达式(placement-new) (了解)

定位new的作用?

        定位new表达式是在已分配的原始内存空间中调用构造函数初始化一个对象


使用格式:

        new (p) type或者new (p) type(initializer-list)

        p必须是一个指针,type(initializer-list)是就是构造函数。意思是在p指向的空间创建一个对象。


使用场景:

        定位new允许开发者在预先分配的内存中构造对象,而不是让new运算符自动在堆上分配内存。定位new表达式在实际中一般是配合内存池使用。因为内存池分配出的内存没有初始化,所以如果是自定义类型的对象,需要使用new的定义表达式进行显示调构造函数进行初始化。

#include <iostream>  
#include <new> // 包含定位new的头文件  
  
class MyClass {  
public:  
    MyClass(int value) : value_(value) {  
        std::cout << "MyClass constructed with value " << value_ << std::endl;  
    }  
    ~MyClass() {  
        std::cout << "MyClass destroyed" << std::endl;  
    }  
    void printValue() const {  
        std::cout << "Value: " << value_ << std::endl;  
    }  
  
private:  
    int value_;  
};  
  
int main() {  
    // 分配足够的内存来存储MyClass对象  
    char buffer[sizeof(MyClass)];  
      
    // 使用定位new在buffer中构造对象  
    MyClass* obj = new (buffer) MyClass(42);  
      
    // 调用对象的成员函数  
    obj->printValue();  
      
    // 手动调用析构函数,因为定位new不会调用delete  
    obj->~MyClass();  
      
    return 0;  
}

9. malloc/free和new/delete的区别

malloc/free和new/delete的共同点是:

  • 都是从堆上申请空间,并且需要用户手动释放。

不同的地方是:

  1. maloc和free是函数,new和delete是操作符。
  2. malloc申请的空间不会初始化,new可以初始化。
  3. malloc的返回值为void*,在使用时必须强转;new不需要,因为new后跟的是空间的类型。
  4. malloc申请空间失败时,返回的是NULL,因此使用时必须判空;new不需要,但是new需要捕获异常。
  5. malloc申请空间时,需要手动计算空间大小并传递,new只需在其后跟上空间的类型即可,如果是多个对象,[ ]中指定对象个数即可。
  6. 申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数与析构函数,而new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间中资源的清理。

------------------------END-------------------------

才疏学浅,谬误难免,欢迎各位批评指正。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

牛客NC242 单词搜索【中等 递归DFS C++/Java/Go/PHP】

题目 题目链接&#xff1a; https://www.nowcoder.com/practice/987f2981769048abaf6180ed63266bb2 思路 递归&#xff1a;以word第一个字符为起点&#xff0c;在矩阵中 递归搜索&#xff0c;检查是否存在完整的word路径&#xff0c; 注意恢复现场&#xff0c;又叫回溯&#…

物联网通信网关的主要功能体现在哪些方面?-天拓四方

在信息化、智能化的时代&#xff0c;物联网技术的广泛应用正在逐渐改变我们的生活方式。物联网通过各种传感器和设备&#xff0c;将现实世界与数字世界紧密相连&#xff0c;从而实现智能化、自动化的生活和工作方式。作为物联网生态系统中的重要组成部分&#xff0c;物联网通信…

MySQL:飞腾2000+Centos7.6 aarch64 部署MySQL8.0.36

目录 1.硬件环境 2.MySQL选择 Bundle版本【全部文件】​编辑 3.下载并安装 4.安装完成后检查mysql 5.初始化MySQL 6.那就问了&#xff0c;都初始化了啥&#xff1f; 7.尝试启动MySQL 8.给mysql文件授权 9.再次尝试启动正常 10.mysql初始化目录出现了mysql.sock 11.找…

VS2022 配置OpenCV开发环境详细教程

OpenCV OpenCV&#xff08;Open Source Computer Vision Library&#xff09;是一个开源的计算机视觉和机器学习软件库&#xff0c;由Intel开发并首先发布于1999年。OpenCV被广泛用于实时图像处理、视频分析、物体检测、面部识别、机器人视觉以及许多其他领域。它支持C、Pytho…

Flutter应用开发-几种保存简单配置的方式

文章目录 简单配置保存的几种方式使用 shared_preferences 插件优点缺点 使用 hive 插件优点 缺点使用文件存储&#xff1a;优点缺点 简单配置保存的几种方式 在 Flutter 开发的 Android 应用中&#xff0c;保存应用配置并下次启动时读取&#xff0c;有以下几种比较合适的方式…

rust疑难杂症解决

rust疑难杂症解决 边碰到边记录&#xff0c;后续可能会逐步增加&#xff0c;备查 cargo build时碰到 Blocking waiting for file lock on package cache 原因是Cargo 无法获取对包缓存的文件锁&#xff0c; 有时vscode中项目比较多&#xff0c;如果其中某些库应用有问题&…

Docker | 入门:安装与配置

Docker | 入门&#xff1a;安装与配置 Docker 和传统虚拟机区别 对于传统虚拟机&#xff1a; 虚拟出一套硬件&#xff0c;运行一个完整的操作系统&#xff0c;并在这个操作系统上安装和运行软件。 对于 Docker: 将一个个容器隔离开。 容器内的应用直接运行在宿主机的内容&am…

软件模型(简洁明了)

《 软件测试基础持续更新中》 一、软件开发模型 1.1 大爆炸模型 优点&#xff1a;思路简单&#xff0c; 通常可能是开发者的“突发奇 想” 缺点&#xff1a;开发过程是非工程化的&#xff0c;随意性大&#xff0c;结果不可预知 测试&#xff1a;开发任务完成后&#xff0c;…

一个自卑的人怎么变得自信

一个自卑的人怎么变得自信 自卑感是一种常见的心理状态&#xff0c;它可能源于个人对自己能力、外貌、价值等方面的负面评价。自卑感不仅会影响一个人的情绪状态&#xff0c;还可能阻碍其在生活、学习和工作中的表现。然而&#xff0c;自信并非一蹴而就的品质&#xff0c;它需要…

基础款:Dockerfile 文件

# bash复制代码# 使用 Node.js 16 作为基础镜像 # 指定一个已经存在的镜像作为模版&#xff0c;第一条必须是from FROM node:16# 将当前工作目录设置为/app # WORKDIR /app# 方法一&#xff1a;用dockerfile命令&#xff1a;进行下载打包文件 # 将 package.json 和 package-loc…

MySQL 之 主从复制

1. 主配置文件&#xff08;win下是my.ini&#xff0c;linux下是my.cnf&#xff09; #mysql 服务ID,保证整个集群环境中唯一 server-id1 #mysql binlog 日志的存储路径和文件名 log-bin/var/lib/mysql/mysqlbin #错误日志,默认已经开启 #log-err #mysql的安装目录 #basedir #mys…

Linux软件包管理器——yum

文章目录 1.什么是软件包1.1安装与删除命令1.2注意事项1.3查看软件包1.3.1注意事项&#xff1a; 2.关于rzsz3.有趣的Linux下的指令 -sl 1.什么是软件包 在Linux下安装软件, 一个通常的办法是下载到程序的源代码, 并进行编译, 得到可执行程序. 但是这样太麻烦了, 于是有些人把一…

穷人想要改命,是选择打工还是创业? 2024创业项目小成本!2024轻资产创业!2024风口行业!2024普通人做什么行业赚钱?

今日话题穷人想要改命&#xff0c;是选择打工还是创业&#xff1f; 改命的方式就是跳进水里&#xff0c;忍受呛水&#xff0c;学会游泳&#xff0c;这个过程越年轻实现越好&#xff0c;就像小鹰往山崖下跳&#xff0c;要么学会飞&#xff0c;要么就狠狠的被摔死。打工思维和创…

用vue3实现留言板功能

效果图&#xff1a; 代码&#xff1a; <script setup lang"ts"> import { ref } from vue;interface Message {name: string;phone: string;message: string; }const name ref<string>(); const phone ref<string>(); const message ref<st…

基于YOLOV5和DeepOCSort的实时目标检测跟踪检测系统

项目简介 本项目旨在研究由YOLOV5模型在多目标检测任务重的应用。通过设计YOLOV5模型及DeepOCSORT模型来实现多物体检测、追踪&#xff0c;最终达高实时性、高精度的物件检测、分割、追踪的效果。最后通过AX620A完成嵌入式硬件部署 项目研究背景 近年来&#xff0c;近年来&am…

【Linux】fork函数详解and写时拷贝再理解

&#x1f490; &#x1f338; &#x1f337; &#x1f340; &#x1f339; &#x1f33b; &#x1f33a; &#x1f341; &#x1f343; &#x1f342; &#x1f33f; &#x1f344;&#x1f35d; &#x1f35b; &#x1f364; &#x1f4c3;个人主页 &#xff1a;阿然成长日记 …

茶叶直播间电商运营带货方案营销计划书

【干货资料持续更新&#xff0c;建议先关注&#xff0c;以防走丢】 茶叶直播间电商运营带货方案营销计划书 部分资料预览 资料部分是网络整理&#xff0c;仅供学习参考。 PPT可编辑&#xff08;完整资料包含以下内容&#xff09; 目录 直播带货方案细化 1. 直播筹划 - 目标…

基于SSM+Jsp+Mysql的汽车租赁系统的设计与实现

开发语言&#xff1a;Java框架&#xff1a;ssm技术&#xff1a;JSPJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包…

OpenHarmony实战开发-如何实现单一手势

点击手势&#xff08;TapGesture&#xff09; TapGesture(value?:{count?:number; fingers?:number})点击手势支持单次点击和多次点击&#xff0c;拥有两个可选参数&#xff1a; count&#xff1a;声明该点击手势识别的连续点击次数。默认值为1&#xff0c;若设置小于1的非…

poi-tl自定义渲染策略学习

文章目录 实现逻辑参考代码注意点 实现逻辑 自定义渲染策略实现逻辑&#xff1a; 找到模板中的表格标签render方法接收java中对应模板表格标签的所有list数据执行自定义渲染逻辑 参考代码 word模板如下&#xff1a; 实体类&#xff1a; Data public class GksxRowData {/…
最新文章