「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.rhkb.cn/news/315620.html

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

相关文章

prime1--vulnhub靶场通关教程

一. 信息收集 1. 探测目标主机IP地址 arp-scan -l //查看网段 vm 编辑--查看虚拟网络编辑器&#xff0c;看到靶机的网段 网段是&#xff1a; 192.168.83.0 是c段网络 2. 全面检测目标IP nmap -sP 192.168.83.1/24 靶机ip是&#xff1a; 192.168.83.145 攻击机的ip是&…

【Java难点】多线程终极

悲观锁和乐观锁 悲观锁 synchronized关键字和Lock的实现类都是悲观锁。 它很悲观&#xff0c;认为自己在使用数据的时候一定有别的线程来修改数据&#xff0c;因此在获取数据的时候会一不做二不休的先加锁&#xff0c;确保数据不会被别的线程修改。 适合写操作多的场景&…

【SpringBoot】00 Maven配置及创建项目

一、Maven配置 1、下载Maven 进入官网下载&#xff1a;Maven – Welcome to Apache MavenMaven – Download Apache Maven 本文以最新版为例&#xff0c;可按需选择版本 Maven – Welcome to Apache Maven 2、解压下载好的安装包 将安装包解压到自己设置的空文件夹中 3、…

7.Prism框架之对话框服务

文章目录 一. 目标二. 技能介绍① 什么是Dialog?② Prism中Dialog的实现方式③ Dialog使用案例一 (修改器)④ Dialog使用案例2(异常显示窗口) 一. 目标 1. 什么是Dialog?2. 传统的Dialog如何实现?3. Prism中Dialog实现方式4. 使用Dialog实现一个异常信息弹出框 二. 技能介…

《HCIP-openEuler实验指导手册》1.3Apache动态功能模块加载卸载练习

1.3.1 配置思路 mod_status 模块可以帮助管理员通过web界面监控Apache运行状态&#xff0c;通过LoadModule指令加载该模块&#xff0c;再配置相关权限&#xff0c;并开启ExtendedStatus后&#xff0c;即可使用该模块。 1.3.2 配置步骤 检查mod_status模块状态&#xff08;使…

C# Solidworks二次开发:访问平面、曲面相关API详解

大家好&#xff0c;今天要介绍的是关于平面、曲面相关的API。 下面是相关的API: &#xff08;1&#xff09;第一个为ISurfacePlanarFeatureData&#xff0c;这个API的含义为允许访问平面表面特征&#xff0c;下面是官方的具体解释&#xff1a; 下面是官方使用的例子&#xff…

【网络原理】IP协议的地址管理和路由选择

系列文章目录 【网络通信基础】网络中的常见基本概念 【网络编程】网络编程中的基本概念及Java实现UDP、TCP客户端服务器程序&#xff08;万字博文&#xff09; 【网络原理】UDP协议的报文结构 及 校验和字段的错误检测机制&#xff08;CRC算法、MD5算法&#xff09; 【网络…

【网络原理】TCP协议的连接管理机制(三次握手和四次挥手)

系列文章目录 【网络通信基础】网络中的常见基本概念 【网络编程】网络编程中的基本概念及Java实现UDP、TCP客户端服务器程序&#xff08;万字博文&#xff09; 【网络原理】UDP协议的报文结构 及 校验和字段的错误检测机制&#xff08;CRC算法、MD5算法&#xff09; 【网络…

扭蛋机小程序带来了什么优势?扭蛋机收益攻略

在当下的潮流消费时代&#xff0c;人们对潮玩也日益个性化&#xff0c;扭蛋机作为一种新型的娱乐消费模式&#xff0c;深受大众喜爱。扭蛋机的价格低&#xff0c;各个年龄层的玩家都可以进行购买&#xff0c;潜在玩家量非常大。扭蛋机商品主打热门IP周边等&#xff0c;种类繁多…

大型零售企业,适合什么样的企业邮箱大文件解决方案?

大型零售企业通常指的是在全球或特定地区内具有显著市场影响力和知名度的零售商。这些企业不仅在零售业务收入上达到了惊人的规模&#xff0c;而且在全球范围内拥有广泛的销售网络和实体店铺。它们在快速变化的零售行业中持续创新&#xff0c;通过实体店、电商平台等多种渠道吸…

第十一章 Spring Boot 整合 WebSocket

第十一章 Spring Boot 整合 WebSocket 1. 为什么需要 WebSocket2. WebSocket 简介3. Spring Boot 整合 WebSocket3.1 实现消息群发1. 依赖2. 配置 WebSocket ************************************************************ 1. 为什么需要 WebSocket 2. WebSocket 简介 3. Spri…

QT支持多种开发语言

QT主要是一个C应用程序框架&#xff0c;但它也提供了对其他一些编程语言的官方或非官方支持。以下是QT支持的一些语言版本及其特点。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流合作。 1.Python (PyQt) &#xff1a; PyQt是QT的官方Pyth…

axios.get请求 重复键问题??

封装的接口方法&#xff1a; 数据&#xff1a; 多选框多选后 能得到对应的数组 但是请求的载荷却是这样的,导致会请求不到数据 departmentChecks 的格式看起来是一个数组&#xff0c;但是通常 HTTP 请求的查询参数不支持使用相同的键&#xff08;key&#xff09;名多次。如…

【Redis 开发】Redis哨兵

哨兵 作用和原理服务状态监控选举新的master 搭建哨兵集群RedisTemplate的哨兵模式 作用和原理 Redis提供了哨兵机制来实现主从集群中的自动故障恢复&#xff1a; 哨兵也是一个集群 监控&#xff1a;会不断检查master和slave是否按预期工作自动故障恢复&#xff1a;如果mast…

本地生活服务平台有哪些?哪个靠谱?

随着多家互联网大厂的本地生活服务布局日益展开&#xff0c;不少人都看到了其中的巨大市场缺口和广阔前景&#xff0c;想要入驻本地生活服务平台&#xff0c;瓜分这块巨大的蛋糕。而在当下这个选择大于努力的时代&#xff0c;能否分到蛋糕以及分到多少蛋糕的关键&#xff0c;就…

Vast+产品展厅 | Vastbase G100数据库是什么架构?(2)

Vastbase G100是海量数据融合了多年对各行业应用场景的深入理解&#xff0c;基于openGauss内核开发的企业级关系型数据库。 上一期&#xff0c;《Vast产品展厅》为您介绍了Vastbase G100的部署架构和物理架构。 本期&#xff0c;我们将为您详细讲解Vastbase G100的物理架构和…

基于Python实现心脏病数据可视化DEA+预测【500010103.1】

一、数据说明 该心脏病数据集是通过组合 5 个已经独立可用但以前未合并的流行心脏病数据集来策划的。在这个数据集中&#xff0c;5 个心脏数据集结合了 11 个共同特征&#xff0c;使其成为迄今为止可用于研究目的的最大心脏病数据集。 该数据集由 1190 个实例和 11 个特征组成…

PVE虚拟机隐藏状态栏虚拟设备

虚拟机启动后&#xff0c;状态栏会出现一些虚拟设备&#xff0c;点击弹出会导致虚拟机无法使用。 解决方案&#xff1a; 1、在桌面新建disable_virtio_removale.bat文件&#xff0c;内容如下&#xff1a; ECHO OFF FOR /f %%A IN (reg query "HKLM\SYSTEM\CurrentContro…

低代码+定制物资管理:创新解决方案探析

引言 在当今快速变化的商业环境中&#xff0c;企业面临着不断增长的挑战&#xff0c;如提高效率、降低成本、满足客户需求等。为了应对这些挑战&#xff0c;企业需要不断创新并采用先进的技术解决方案。在这样的背景下&#xff0c;低代码开发和定制化物资管理成为了引领企业变…

Spark-机器学习(7)分类学习之决策树

在之前的文章中&#xff0c;我们学习了分类学习之支持向量机&#xff0c;并带来简单案例&#xff0c;学习用法。想了解的朋友可以查看这篇文章。同时&#xff0c;希望我的文章能帮助到你&#xff0c;如果觉得我的文章写的不错&#xff0c;请留下你宝贵的点赞&#xff0c;谢谢。…