C/C++内存管理(含C++中new和delete的使用)

文章目录

  • C/C++内存管理(含C++中new和delete的使用)
    • 1、C/C++内存分布
    • 2、C语言中动态内存管理方式:malloc/calloc/realloc/free
    • 3、C++动态内存管理
      • 3.1、new/delete操作内置类型
      • 3.2、new/delete操作自定义类型
    • 4、operator new与operator delete函数
    • 5、new和delete的实现原理
      • 5.1、内置类型
      • 5.2、自定义类型
    • 6、定位new表达式(placement-new)

img

C/C++内存管理(含C++中new和delete的使用)

1、C/C++内存分布

我们先来看下面的一段代码和相关问题。

int globalVar = 1;
static int staticGlobalVar = 1;int main() {static int staticVar = 1;int localVar = 1;int num1[10] = {1, 2, 3, 4};char char2[] = "abcd";const char *pChar3 = "abcd";int *ptr1 = (int *) malloc(sizeof(int) * 4);int *ptr2 = (int *) calloc(4, sizeof(int));int *ptr3 = (int *) realloc(ptr2, sizeof(int) * 4);free(ptr1);free(ptr3);return 0;
}
//1. 选择题:
// 选项: A.栈 B.堆 C.数据段(静态区) D.代码段(常量区)
// globalVar在哪里?____  staticGlobalVar在哪里?____
// staticVar在哪里?____  localVar在哪里?____
// num1 在哪里?____
//
// char2在哪里?____  *char2在哪里?___
// pChar3在哪里?____   *pChar3在哪里?____
// ptr1在哪里?____ *ptr1在哪里?____
//2. 填空题:
// sizeof(num1) = ____; 
//sizeof(char2) = ____;  strlen(char2) = ____;
// sizeof(pChar3) = ____;   strlen(pChar3) = ____;
// sizeof(ptr1) = ____;
//3. sizeof 和 strlen 区别?
//

  • 说明
    1. 又叫堆栈–非静态局部变量/函数参数/返回值等等,栈是向下增长的。
    2. 内存映射段 是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享共享内存,做进程间通信。(Linux课程如果没学到这块,现在只需要了解一下)
    3. 用于程序运行时动态内存分配,堆是可以上增长的。
    4. 数据段存储全局数据和静态数据
    5. 代码段可执行的代码/只读常量

所以1.选择题有答案了。

那么2.填空题答案是:之前学过不讲了哈哈。

其中3.sizeofstrlen 区别?

答:strlen遇到\0结束,strlen算字符串长度,sizeof算变量大小。


2、C语言中动态内存管理方式:malloc/calloc/realloc/free

malloccallocrealloc 是在C语言中用于动态内存分配的三个函数,它们有一些区别,主要体现在它们的功能和用法上。

  1. malloc(Memory Allocation,内存分配):

    • malloc 是 “memory allocation” 的缩写,用于分配指定大小的内存块。
    • 它只分配内存,不对内存进行初始化,所以分配的内存中可能包含任意值。
    void* malloc(size_t size);
    
  2. calloc(Contiguous Allocation,连续分配):

    • calloc 也是用于分配内存的函数,但与 malloc 不同的是,calloc 分配的内存块会被初始化为零
    • calloc 接受两个参数,分别是所需的元素个数和每个元素的大小。
    void* calloc(size_t num_elements, size_t element_size);
    
  3. realloc(Re-allocation,重新分配):

    • realloc 用于重新分配已分配内存的大小,可以用于扩大或缩小内存块的大小。
    • 如果原始内存块的地址不为空,realloc 会尝试在原始地址上修改内存块的大小(如果新内存大小大于这块原始空间,则还是需要重新开辟内存),如果原始地址为空,则行为类似于 malloc
    void* realloc(void* ptr, size_t new_size);
    

总结:

  • malloc 只分配指定大小的内存块,不进行初始化。
  • calloc 分配指定数量和大小的内存块,并将内存块的所有位初始化为零。
  • realloc 重新分配内存块的大小,可以用于扩大或缩小已分配内存的大小。
int main() {int *p1 = (int *) malloc(sizeof(int));free(p1);
// 1.malloc/calloc/realloc的区别是什么?int *p2 = (int *) calloc(4, sizeof(int));int *p3 = (int *) realloc(p2, sizeof(int) * 10);
// 这里需要free(p2)吗? --- 看情况free(p3);return 0;
}

3、C++动态内存管理

C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力(比如给对象初始化),而且使用起来比较麻烦,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理

3.1、new/delete操作内置类型

int main() {// 动态申请一个int类型的空间int *ptr4 = new int;// 动态申请一个int类型的空间并初始化为10int *ptr5 = new int(10);cout << *ptr5 << endl;// 动态申请10个int类型的空间int *ptr6 = new int[3];// 动态申请10个int类型的空间并初始化前3个int *ptr7 = new int[3]{1, 2, 3};cout << ptr7[0] << ptr7[1] << ptr7[2] << ptr7[3];delete ptr4;delete ptr5;delete[] ptr6;delete[] ptr7;
}
  • 初始化格式
    • 单个元素空间:new 类型 (初始化值)
    • 连续空间:new 类型 [元素个数] {从前往后元素初始化值,其余元素初始化为0}
  • 注意:申请和释放单个元素的空间,使用newdelete操作符,申请和释放连续的空间,使用new[]delete[]。注意:匹配起来使用

3.2、new/delete操作自定义类型

class Date {
public:Date() : _year(1), _month(1), _day(1) {cout << "Date()" << endl;}Date(int year, int month, int day) : _year(1), _month(1), _day(1) {_year = year;_month = month;_day = day;cout << "Date()" << endl;}~Date() {cout << "~Date()" << endl;}private:int _year;int _month;int _day;
};int main() {Date *ptr1 = new Date();Date *ptr2 = new Date(2, 2, 2);Date *ptr3 = new Date[10]{{1, 2, 2}};free(ptr1);delete ptr2;delete[] ptr3;return 0;
}

  • 初始化格式
    • 单个元素空间:new 类型 (初始化值)
    • 连续空间:new 类型 [元素个数] {从前往后元素初始化值使用{}代表每一个元素的值,其余元素初始化为0}
  • new/deletemalloc/free最大区别是 new/delete对于【自定义类型】除了开空间还会调用构造函数和析构函数

4、operator new与operator delete函数

newdelete是用户进行动态内存申请和释放的操作符,operator newoperator delete是系统提供的全局函数,new底层调用operator new全局函数来申请空间,delete底层通过operator delete全局函数来释放空间。

  • operator new:该函数实际通过malloc来申请空间,当malloc申请空间成功时直接返回;申请空间失败,尝试执行空间不足应对措施,如果改应对措施用户设置了,则继续申请,否则抛异常。
  • operator delete 最终是通过free来释放空间的。
class Date {
public:Date() : _year(1), _month(1), _day(1) {cout << "Date()" << endl;}Date(int year, int month, int day) : _year(1), _month(1), _day(1) {_year = year;_month = month;_day = day;cout << "Date()" << endl;}~Date() {cout << "~Date()" << endl;}private:int _year;int _month;int _day;
};int main() {//operator new -- 不调用构造函数 和 malloc 基本一样Date *ptr6 = (Date *) operator new(sizeof(Date));delete new(ptr6) Date;ptr6 = nullptr;return 0;
}

5、new和delete的实现原理

5.1、内置类型

如果申请的是内置类型的空间,newmallocdeletefree基本类似,不同的地方是:new/delete申请和释放的是单个元素的空间,new[]delete[]申请的是连续空间,而且new在申请空间失败时会抛异常malloc会返回NULL

5.2、自定义类型

  • new的原理

    1. 调用operator new函数申请空间
    2. 在申请的空间上执行构造函数,完成对象的构造
  • delete的原理

    1. 在空间上执行析构函数,完成对象中资源的清理工作
    2. 调用operator delete函数释放对象的空间

  • new T[N]的原理

    1. 调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申请

    2. 在申请的空间上执行N次构造函数

  • delete[]的原理

    1. 在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理

    2. 调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释放空间

这里我们需要注意一个现象对于内置类型在new T[N]时候,往往开辟的内存空间会大于N*sizeof(T),可能会多出4个字节

原本这里ptr2开辟的空间size应该是12*10 = 120,但是这里显示124,其中4个字节是用来存储开辟连续Date对象的个数。

class Date {
public:Date() : _year(1), _month(1), _day(1) {cout << "Date()" << endl;}Date(int year, int month, int day) : _year(1), _month(1), _day(1) {_year = year;_month = month;_day = day;cout << "Date()" << endl;}~Date() {cout << "~Date()" << endl;}private:int _year;int _month;int _day;
};int main() {Date* ptr1 = new Date;delete ptr1;Date* ptr2 = new Date[10];delete[] ptr2;return 0;
}

但是如果是内置类型new T[N],则不需要另外的空间存个数

原本这里ptr3开辟的空间size应该是4*10 = 40,并且这里显示40,即没有开辟空间存个数。

class Date {
public:Date() : _year(1), _month(1), _day(1) {cout << "Date()" << endl;}Date(int year, int month, int day) : _year(1), _month(1), _day(1) {_year = year;_month = month;_day = day;cout << "Date()" << endl;}~Date() {cout << "~Date()" << endl;}private:int _year;int _month;int _day;
};int main() {Date* ptr1 = new Date;delete ptr1;Date* ptr2 = new Date[10];delete[] ptr2;int* ptr3 = new int[10];delete[] ptr3;return 0;
}

原因:这多出来的4个字节是用来记录开辟T大小连续空间的个数,以便于delete [] 进行析构和释放空间。

这里如果不写析构函数(默认成员变量没有开辟空间,如果开辟了空间,必须调用析构函数释放空间,不然会内存泄露)的话就不报错,因为成员变量都是内置类型没有开空间,不需要调用析构函数,也就不需要在前面添加4字节存个数,即自定义类型new T[N]可以直接delete,内置类型也一样。

class Date {
public:Date() : _year(1), _month(1), _day(1) {cout << "Date()" << endl;}Date(int year, int month, int day) : _year(1), _month(1), _day(1) {
//        _a = new int[10];_year = year;_month = month;_day = day;cout << "Date()" << endl;}//    ~Date() {
//        cout << "~Date()" << endl;
//    }private:
//    int* _a;int _year;int _month;int _day;
};int main() {Date *ptr1 = new Date;delete ptr1;Date *ptr2 = new Date[10];// 这里如果不写析构函数(默认成员变量没有开辟空间,如果开辟了空间,必须调用析构函数释放空间,不然会内存泄露)的话就不报错// 因为成员变量都是内置类型没有开空间,不需要调用析构函数,也就不需要在前面添加4字节存个数delete ptr2;return 0;
}

原本这里ptr2开辟的空间size应该是12*10 = 120,并且这里显示120,即没有开辟空间存对象个数。


6、定位new表达式(placement-new)

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

  • 使用格式new (place_address) type或者new (place_address) type(initializer-list)
    • place_address必须是一个指针,initializer-list是类型的初始化列表
  • 使用场景
    • 定位new表达式在实际中一般是配合内存池使用。因为内存池分配出的内存没有初始化,所以如果是自定义类型的对象,需要使用new的定义表达式进行显示调构造函数进行初始化。
class Date {
public:Date() : _year(1), _month(1), _day(1) {cout << "Date()" << endl;}Date(int year, int month, int day) : _year(1), _month(1), _day(1) {_year = year;_month = month;_day = day;cout << "Date()" << endl;}~Date() {cout << "~Date()" << endl;}private:int _year;int _month;int _day;
};int main() {Date *p = (Date *) operator new(sizeof(Date));// 不能显式调用构造函数// p->Date();// 定位new可以显式调用构造函数new(p)Date(1, 1, 1);p->~Date();return 0;
}

OKOK,C/C++内存管理就到这里。如果你对Linux和C++也感兴趣的话,可以看看我的主页哦。下面是我的github主页,里面记录了我的学习代码和leetcode的一些题的题解,有兴趣的可以看看。

Xpccccc的github主页

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

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

相关文章

ChatGPT到底是如何运作?

自从2022年11月30日发布以来&#xff0c;ChatGPT一直占据着科技届的头条位置&#xff0c;随着苹果的创新能力下降&#xff0c;ChatGPT不断给大家带来震撼&#xff0c;2023年11月7日&#xff0c;首届OpenAI开发者大会在洛杉矶举行&#xff0c;业界普遍认为&#xff0c;OpenAI的开…

走近科学之《MySQL 的秘密》

走近科学之《MySQL 的秘密》 mysql 存储引擎、索引、执行计划、事务、锁、分库分表、优化 1、存储引擎&#xff08;storage engines&#xff09; 存储引擎规定了数据存储时的不同底层实现&#xff0c;如存储机制、索引、锁、事务等。 可以通过 show engines 命令查看当前服务…

【JavaSE】:接口(二)

接口 一.对学生进行排序二.Clonable接口三.抽象类和接口的区别四.Object类 一.对学生进行排序 很明显我们直接对学生进行排序是会报错的&#xff0c;因为编译器也不知道你需要按照名字排还是分数排&#xff0c;我们点击报错位置进入sort内部。 在sort内部我们可以看到它把数组转…

泛微OA对接金蝶云星空方案分享(对接场景解析)

分享金蝶云星空跟泛微OA系统集成对接的方案分享&#xff0c;主讲审批流程对接&#xff0c;表单对接的两类场景。分别是金蝶云星空发起申请和泛微发起流程审批&#xff0c;最终实现统一管理。 数据集成主要有以下好处&#xff1a; &#xff08;1&#xff09;数据一致性&#xf…

东南大学与OpenHarmony携手共建开源生态,技术俱乐部揭牌成立并迎来TSC专家进校园

11月25日,OpenAtom OpenHarmony(以下简称“OpenHarmony”)项目群技术指导委员会(以下简称“TSC”)与东南大学携手,于东南大学九龙湖校区金智楼一楼报告厅举办了“东南大学OpenHarmony技术俱乐部成立仪式暨OpenHarmony TSC专家进校园”活动。此次盛会标志着OpenHarmony开源社区和…

Mac M1 安装Docker打包arm64的python项目的镜像包

1、首先安装Docker&#xff0c;到官网下载&#xff0c;选择apple chip版 Docker中文网 官网 2、双击下载的dmg文件&#xff0c;在弹出框中之间拖拽到右边 3、打开docker&#xff0c;修改国内镜像源&#xff0c;位置在配置-DockerEngine "registry-mirrors": ["…

Go字符串类型

一、字符串 1、字符串 Go 语言里的字符串的内部实现使用 UTF-8 编码字符串带的值为双引号&#xff08;"&#xff09;中的内容&#xff0c;可以在 Go 语言的源码中直接添加非ASCII 码字符 s1 : "hello" s2 : "您好" 2、字符串转义符 Go 语言的字符…

UE4 UE5 使用SVN控制

关键概念&#xff1a;虚幻引擎中使用SVN&#xff0c;帮助团队成员共享资源。 1. UE4/UE5项目文件 如果不需要编译的中间缓存&#xff0c;则删除&#xff1a; DerivedDataCache、Intermediate、Saved 三个文件夹 2.更新、上传

【linux】基本指令(中篇)

echo指令 将引号内容打印到显示屏上 输出的重定向 追加的重定向 输出的重定向 我们学习c语言的时候当以写的方式创建一个文件&#xff0c;就会覆盖掉该文件之前的内容 当我们以追加的方式打开文件的时候&#xff0c;原文件内容不会被覆盖而是追加 more指令 10.more指令…

YOLOv8独家原创改进:自研独家创新MSAM注意力,通道注意力升级,魔改CBAM

💡💡💡本文自研创新改进:MSAM(CBAM升级版):通道注意力具备多尺度性能,多分支深度卷积更好的提取多尺度特征,最后高效结合空间注意力 1)作为注意力MSAM使用; 推荐指数:五星 MSCA | 亲测在多个数据集能够实现涨点,对标CBAM。 在道路缺陷检测任务中,原始ma…

基于51单片机超声波测距汽车避障系统

**单片机设计介绍&#xff0c; 基于51单片机超声波测距汽车避障系统 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于51单片机的超声波测距汽车避障系统是一种用于帮助汽车避免碰撞和发生事故的设备&#xff0c;以下是一个基本…

Java面试-框架篇-Mybatis

Java面试-框架篇-Mybatis MyBatis执行流程延迟加载使用及原理一, 二级缓存来源 MyBatis执行流程 读取MyBatis配置文件: mybatis-config.xml加载运行环境和映射文件构造会话工厂SqlSessionFactory会话工厂创建SqlSession对象(包含了执行SQL语句的所有方法)操作数据库的接口, Ex…

从零开始学习typescript——数据类型

数据类型 以前我们用js编写代码的时候&#xff0c;都是直接使用let、var、const 来定义数据类型&#xff1b;js会在运行时来确定数据类型&#xff0c;但是在ts中&#xff0c;可以在声明时就可以指定数据类型。如果你学过其他编程语言&#xff0c;比如c、java就能更好的理解了。…

HTML5原生视频播放器组件video的videocontrolslist属性详解

HTML5提供了内置的视频播放控件,其中videocontrolslist是其中一个很有用的属性。videocontrolslist属性可以用于告诉浏览器在视频播放过程中应该显示哪些默认的用户界面控件。下面我们将从几个方面来介绍videocontrolslist的详细使用。 一、启用videocontrolslist videocont…

格式工厂——万能格式转换器

很多时候&#xff0c;大家从网络上下载的文件不一定是自己想要的类型&#xff0c;比如flv等视频文件&#xff0c;而强行改文件后缀名只会造成文件格式错误&#xff0c;无法打开&#xff1b;而很多文件的格式并不能轻易转换&#xff0c;且很多渠道都需要收费。今天介绍的这款For…

实力登榜!迅镭激光荣膺“江苏省瞪羚企业”称号!

11月24日&#xff0c;江苏省生产力促进中心发布“江苏省独角兽企业和瞪羚企业评估结果”&#xff0c;迅镭激光凭借在技术研发、产品创新、市场服务等方面的取得的领先成就&#xff0c;荣膺“江苏省瞪羚企业”称号! “瞪羚企业”是指以科技创新或商业模式创新为支撑&#xff0c;…

一、Gradle 手动创建一个项目

文章目录 Gradle 介绍Gradle Wrapper Gradle 使用手动安装 Gradle初始化 Gradle 介绍 Gradle 是一个快速的、可信的、适应性强的自动化构建工具&#xff0c;它是开源的。它使用优雅的并且可扩展的描述性语言。其他的介绍在官网可以了解。 Gradle Wrapper 官方建议使用 Gradl…

c语言:模拟实现各种字符串函数

strlen函数&#xff1a; 功能&#xff1a;获取到\0之前的的字符个数。 代码模拟实现函数&#xff1a; //strlen //这里用了递归法&#xff0c; //如abc&#xff0c;1bc&#xff0c;然后11c&#xff0c;接着111&#xff0c;最后读取到\0&#xff0c;1110&#xff0c;得到结果3…

Day40力扣打卡

打卡记录 包子凑数&#xff08;裴蜀定理 DP&#xff09; 根据裴蜀定理&#xff0c;存在 c gcd(a, b) 使不定方程ax by c满足条件&#xff0c;如果gcd(a, b) 1即a与b互素的情况下&#xff0c;就会 ax by 1&#xff0c;由于为1可以构造后面的无穷数字&#xff0c;故得到结…

深入了解Java8新特性-日期时间API之ZonedDateTime类

阅读建议 嗨&#xff0c;伙计&#xff01;刷到这篇文章咱们就是有缘人&#xff0c;在阅读这篇文章前我有一些建议&#xff1a; 本篇文章大概19000多字&#xff0c;预计阅读时间长需要10分钟以上。本篇文章的实战性、理论性较强&#xff0c;是一篇质量分数较高的技术干货文章&…