【C\C++】内存分配 和 动态内存管理方式

news2025/1/11 14:07:16

文章目录

  • 内存分类
    • 题目:知识巩固
      • 选择题: 变量位于内存中的位置
      • 计算题 变量值的大小
    • 答案
  • C语言 动态内存管理
    • malloc / calloc / realloc
      • 作用
      • 区别
  • C++ 内存管理方式
    • operator new 与 operator delete
    • new 与 delete 的实现原理
    • malloc free 与 new delete 的区别
  • 内存泄漏

内存分类

在C++中,有几个重要的内存区域,每个区域都有不同的意义和用途。我们从内存分配的角度来分析C++各个内存区域的含义:

  1. 栈(Stack):栈是用于存储 局部变量、函数参数以及函数调用信息 的内存区域。它的特点是自动分配和释放,并且遵循后进先出的原则(LIFO)。

栈的大小有限,通常比较小,因此栈上的变量不能太大。

当一个函数被调用时,它的局部变量和参数将在栈上分配内存。当函数返回时,这些内存将自动释放。


  1. 堆(Heap):堆是用于 动态分配内存 的区域。通过 “new” 或 “malloc” 等操作符可以在堆上分配内存,并使用指针来访问和操作这块内存。

堆上分配的内存需要手动释放,否则可能导致内存泄漏。堆的大小通常比栈大得多,但也受到操作系统的限制。


  1. 全局/静态存储区(Global/Static Storage Area):全局存储区用于存储 全局变量和静态变量 。全局变量具有程序的整个生命周期,而静态变量的生命周期与其作用域相对应。

全局/静态存储区在程序启动时分配,直到程序结束才会释放。


  1. 常量存储区(Constant Storage Area):常量存储区用于存储 常量数据 ,如字符串常量。这些数据是只读的,无法修改。

常量存储区通常位于静态存储区中。


  1. 程序代码区(Code Section):程序代码区存储了 程序的执行指令 。这个区域通常是只读的,包含了可执行文件的机器指令

代码区也叫做文本区。


需要注意的是,内存区域的名称和具体实现可能因编译器、操作系统或平台而异。上述区域的描述是一般情况下的概念,可以帮助我们理解C++程序中内存的分配和使用方式。


题目:知识巩固

根据上述的分类定义,看下面一道经典的题来巩固知识:
在这里插入图片描述


选择题: 变量位于内存中的位置

A.栈 B.堆 C.数据段(静态区) D.代码段(常量区)

变量位置变量位置
globalVar~~~staticGlobal~~~
staticVal~~~localVar~~~
num1~~~
char2~~~*char2~~~
pChar3~~~*pChar3~~~
ptr1~~~*ptr1~~~

计算题 变量值的大小

sizeofstrlen
sizeof(num1)
sizeof((char2)strlen(char2)
sizeof((pChar2)strlen(pChar3)
sizeof(ptr1)

答案

根据上述代码,可以完善表格:
在这里插入图片描述
A.栈 B.堆 C.数据段(静态区) D.代码段(常量区)

变量位置变量位置
globalVarCstaticGlobalC
staticValClocalVarA
num1A
char2A*char2A
pChar3A*pChar3D
ptr1A*ptr1B

*pChar3 的解释:
在C语言中,字符串常量如"abcd"通常被视为字面常量,在编译时存储在代码段中。指针变量pChar3保存着字符串常量"abcd"的首地址,即代码段中字符串的起始位置。

对于上述变量 的 大小/长度 的计算结果:

sizeofstrlen
sizeof(num1) = 4
sizeof((char2) = 5strlen(char2) = 4
sizeof((pChar2) = 4 / 8strlen(pChar3) = 4
sizeof(ptr1) = 4 / 8

C语言 动态内存管理

在C语言中,动态内存管理是通过以下四个函数来实现的:

  • malloc()
  • calloc()
  • realloc()
  • free()

malloc / calloc / realloc

作用

mallocmalloc函数用于在堆区分配指定大小的内存。
它接受一个参数,即要分配的字节数。如果分配成功,返回一个指向分配内存起始地址的指针;如果分配失败,则返回NULL。malloc分配的内存不会被初始化,它的内容是未定义的。

calloccalloc函数也用于在堆区分配内存,但与malloc不同的是,它还会将分配的内存块全部初始化为零
。它需要两个参数,即要分配的元素数量和每个元素的大小。calloc的返回值是一个指向分配内存起始地址的指针。如果分配成功,返回的内存将被清零;如果分配失败,则返回NULL。

realloc:realloc函数用于调整之前通过malloc或calloc分配的内存的大小。它接受两个参数,一个是之前分配内存的指针,另一个是新的内存大小。realloc会尝试重新分配指定大小的内存,并将之前分配的数据复制到新的内存中。如果分配成功,返回新的内存地址;如果分配失败,则返回NULL。如果传递给realloc的指针为NULL,其行为等同于malloc。

区别

主要区别

  • malloc 只负责分配内存,并且不会对分配的内存进行初始化
  • calloc 除了分配内存,还会将分配的内存块清零
  • realloc 用于重新调整内存大小,并且可以在原有内存块中保留数据。需要注意的是,使用realloc重新分配内存时,不能保证原有内存块的内容被保留在同一位置,因此在使用realloc后,要谨慎处理原有指针的引用。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main() {
    int* ptr1;
    int* ptr2;
    int* ptr3;

    // 使用malloc分配内存
    ptr1 = (int*)malloc(5 * sizeof(int));
    if (ptr1 == NULL) {
        printf("内存分配失败!\n");
        return 1;
    }

    printf("使用malloc分配内存后的初始值:\n");
    for (int i = 0; i < 5; i++) {
        printf("%d ", ptr1[i]);
    }
    printf("\n\n");

    // 使用calloc分配内存
    ptr2 = (int*)calloc(5, sizeof(int));
    if (ptr2 == NULL) {
        printf("内存分配失败!\n");
        return 1;
    }

    printf("使用calloc分配内存后的初始值:\n");
    for (int i = 0; i < 5; i++) {
        printf("%d ", ptr2[i]);
    }
    printf("\n\n");

    // 使用realloc重新调整内存大小,并手动初始化新增内存为零
    ptr3 = (int*)realloc(ptr2, 10 * sizeof(int));
    if (ptr3 == NULL) {
        printf("内存分配失败!\n");
        return 1;
    }

    // 将新增的内存空间初始化为零
    memset(ptr3 + 5, 0, 5 * sizeof(int));

    printf("使用realloc重新调整内存大小后的初始值:\n");
    for (int i = 0; i < 10; i++) {
        printf("%d ", ptr3[i]);
    }
    printf("\n\n");

    free(ptr1);
    free(ptr3);

    return 0;
}

输出结果:

使用malloc分配内存后的初始值:
-842150451 -842150451 -842150451 -842150451 -842150451

使用calloc分配内存后的初始值:
0 0 0 0 0

使用realloc重新调整内存大小后的初始值:
0 0 0 0 0 0 0 0 0 0

从输出结果可以看出:

  • 使用 malloc 分配的内存没有被初始化,其内容是未定义的;
  • 使用 calloc 分配的内存被自动初始化为零
  • 使用 realloc 重新调整内存大小后新内存中的内容是不确定的,但之前的数据被保留在新内存块中。

C++ 内存管理方式

栈: 栈是一块自动管理的内存区域 ,用于存储局部变量和函数调用的上下文信息。在函数调用时,其局部变量会被自动分配在栈上。当函数返回时,这些局部变量会自动被销毁,释放相应的内存空间。 栈上分配的内存管理非常高效,但是分配的内存空间大小是固定的,且生命周期随函数的调用而限制。

栈的模拟实现

堆: 堆是一块动态管理的内存区域,用于存储动态分配的对象 。在C++中,**使用new关键字从堆上分配内存空间,动态创建对象。并使用delete来进行手动进行释放资源。

堆的模拟实现

智能指针: 智能指针是一种包装类 可以像一般指针一样访问对象,但内部会自动管理生命周期,当不再需要时会自动释放内存。

智能指针

容器类: C++标准库提供了各种容器类 ,如 std::vector、std::list 等,这些容器类可以自动管理内存分配和释放。它们会自动在堆上分配所需的内存,并在对象生命周期结束时自动释放相应的内存


operator new 与 operator delete

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

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

void *__CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc)
{
	// try to allocate size bytes
	void *p;
	while ((p = malloc(size)) == 0)
	if (_callnewh(size) == 0)
	{
		// report no memory
		// 当申请内存失败了,会抛出 bad_alloc 类型异常
		static const std::bad_alloc nomem;
		_RAISE(nomem);
	}
	return (p);
}
/*
operator delete: 该函数最终是通过free来释放空间的
*/

void operator delete(void *pUserData)
{
	_CrtMemBlockHeader * pHead;
	RTCCALLBACK(_RTC_Free_hook, (pUserData, 0));
	if (pUserData == NULL)
	return;
	_mlock(_HEAP_LOCK); /* block other threads */
	__TRY
	/* get a pointer to memory block header */
	pHead = pHdr(pUserData);
	/* verify block type */
	_ASSERTE(_BLOCK_TYPE_IS_VALID(pHead->nBlockUse));
	_free_dbg( pUserData, pHead->nBlockUse );
	__FINALLY
	_munlock(_HEAP_LOCK); /* release other threads */
	__END_TRY_FINALLY
	return;
}
/*
free的实现
*/
#define free(p) _free_dbg(p, _NORMAL_BLOCK)

从上面的代码(两个全局函数的实现)可以看出:
operatornew 实际也是通过malloc来申请空间,如果malloc申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供该措施就继续申请,否则就抛异常。
operator delete 最终是通过free来释放空间的


new 与 delete 的实现原理

  • 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来释放空间

malloc free 与 new delete 的区别

malloc/free和new/delete的共同点是: 都是从堆上申请空间,并且需要用户手动释放 。不同的地方由下面的表格列出:

mallocnew
函数操作符
申请空间不初始化初始化
申请空间时,需要手动计算空间大小并传递new只需在其后跟上空间的类型,多个对象,[]中指定对象个数即可
返回值为void*, 在使用时需要强转不需要强转,new后跟的是空间的类型
申请空间失败时返回NULL,需要判空不需要判空,需要捕获异常

malloc - freenew - delete
申请自定义类型对象时,malloc/free仅开辟空间,不调用构造函数与析构函数new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间中资源的清理

内存泄漏

简单了解内存泄漏

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

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

相关文章

MySql 变量

1.系统变量 1.1 系统变量分类 变量由系统定义&#xff0c;不是用户定义&#xff0c;属于 服务器 层面。系统变量分为全局系统变量&#xff08;需要添加 global 关键字&#xff09;以及会话系统变量&#xff08;需要添加 session 关键字&#xff09;&#xff0c;有时也把全局系…

mysql UUID 作为主键的问题

UUID 在MySQL中&#xff0c;可以使用UUID()函数来生成一个新的UUID值。该函数的返回值是一个字符串类型&#xff0c;表示一个32位的十六进制数字&#xff0c;其中包含4个连字符“-”&#xff0c;例如&#xff1a;“6ccd780c-baba-1026-9564-0040f4311e29”。 varchar(32) 32*4…

【免费模板】2023数学建模国赛word+latex模板免费分享

无需转发 免费获取2023国赛模板&#xff0c;获取方式见文末 模板文件预览如下&#xff1a; 模板参考格式如下&#xff1a; &#xff08;题目&#xff09;XXXXXX 摘 要&#xff1a; 开头段&#xff1a;需要充分概括论文内容&#xff0c;一般两到三句话即可&#xff0c;长度控…

Java基础二十五(Map)

Map 接口 Map 接口是 Java 集合框架中的一种用于储存键值对映射关系的接口。Map 接口提供了一种通过键来访问值的方式&#xff0c;其中每一个键都是唯一的&#xff0c;值可以重复。 public interface Map<K,V>Map 接口的主要特征如下&#xff1a; 键唯一性&#xff1a;…

matlab数据处理: cell table array+datetime

原数据文件.csv matlab xlsread(filename{i},B2:T2881) 会同于Excel最多1048576行 舍弃 a{1,i} xlsread(filename{i},‘B2:T2881’);%读取excel文件,选定区域’B2:G2881’ readcell(filename{i},Range,E2:M2881) 会全部读取 优选 对于日期 yyyy-MM-dd HH:mm:ss.000 matlab cel…

开始MySQL之路——MySQL三大日志(binlog、redo log和undo log)概述详解

前言 MySQL实现事务、崩溃恢复、集群的主从复制&#xff0c;底层都离不开日志&#xff0c;所以日志是MySQL的精华所在。只有了解MySQL日志&#xff0c;才算是彻底搞懂MySQL。 日志是mysql数据库的重要组成部分&#xff0c;记录着数据库运行期间各种状态信息。mysql日志主要包…

【Linux从入门到精通】通信 | 管道通信(匿名管道 命名管道)

本派你文章主要是对进程通信进行详解。主要内容是介绍 为什么通信、怎么进行通信。其中本篇文章主要讲解的是管道通信。希望本篇文章会对你有所帮助。 文章目录 一、进程通信简单介绍 1、1 什么是进程通信 1、2 为什么要进行通信 1、3 进程通信的方式 二、匿名管道 2、1 什么是…

基于单片机的八路抢答器(数码管版)(独立按键、四位共阳极数码管、指示灯)

随着科学技术的发展和普及&#xff0c;各种各样的竞赛越来越多&#xff0c;其中抢答器的作用也就显而易见。目前很多抢答器基本上采用小规模数字集成电路设计&#xff0c;使用起来不够理想。因此设计一更易于使用和区分度高的抢答器成了非常迫切的任务。现在单片机已进入各个领…

Python UI自动化 —— pytest常用运行参数解析、pytest执行顺序解析

pytest常用Console参数&#xff1a; -v 用于显示每个测试函数的执行结果-q 只显示整体测试结果-s 用于显示测试函数中print()函数输出-x 在第一个错误或失败的测试中立即退出-m 只运行带有装饰器配置的测试用例-k 通过表达式运行指定的测试用例-h 帮助 首先来看什么参数都没加…

QT 第四天

一、设置一个闹钟 .pro QT core gui texttospeechgreaterThan(QT_MAJOR_VERSION, 4): QT widgetsCONFIG c11# The following define makes your compiler emit warnings if you use # any Qt feature that has been marked deprecated (the exact warnings # depend…

RHCE——十八、shell编程之sed

RHCE 一、概念工作原理 二、基本语法1、格式2、参数3、定址符4、操作 三、输出文本范例文件 四、文本替换1、范例文件2、格式3、示例4、使用替换实现删除 五、删除文本1、注意2、示例 六、插入文本1、注意2、格式3、示例4、注意 七、练习 一、概念 sed命令是一个非交互式的文本…

[keil] uv编译分析

假设Keil安装路径: C:\Keil_v5\ 假设工程在 d:\HELLO , 工程Targets名:Simulator [在Manage Project Items中可修改] 如下指令为:Build(F7) C:\Keil_v5\UV4\UV4.exe -b d:\HELLO\Hello.uvproj -j0 -t Simulator -o d:\HELLO\uv4.log 如下指令为:Rebuild(CtrlAltF7) C:\Kei…

PEX装机

目录 一、PXE是什么&#xff1f; 二、PXE的组件&#xff1a; vsftpd/httpd/nfs tftp dhcp 三、配置vsftpd 四、配置tftp 1.安装tftp-server 2.启动tftp 五、准备pxelinx.0文件、引导文件、内核文件 1.准备pxelinux.0文件 2.准备引导文件、内核文件 六、配置dhcp …

【基础建设】浅谈企业网络安全运营体系建设

引言 在网络安全环境复杂又严峻的当前&#xff0c;国内各大企业已开始组建自己的网络安全团队&#xff0c;加强企业自身安全能力建设&#xff0c;朝着网络安全运营一体化迈进。但企业安全运营也已逐步从被动式转变为主动式&#xff0c;成为将人、管理与技术结合&#xff0c;全…

C高级day1 shell 指令的补充学习

使用cut截取出Ubuntu用户的家目录&#xff0c;要求&#xff1a;不能使用":"作为分割 2.思维导图

【WPF C#】PorphyStruct类卟啉和类咕啉的结构分析

前言 首先&#xff0c;热烈祝贺家姐申请到了国家自然科学基金&#xff08;8月底&#xff09;&#xff0c;找一些化学领域的程序和软件&#xff0c;助我姐一臂之力&#xff0c;顺便自己研究一下源码。 卟啉类化合物的结构分析 PorphyStruct&#xff0c;一种用于分析不同卟啉类非…

tlog实现链路追踪

tlog实现链路追踪 TLog通过对日志打标签完成企业级微服务的日志追踪。它不收集日志&#xff0c;使用简单&#xff0c; 产生全局唯一追踪码。除了追踪码&#xff0c;TLog还支持SpanId和上下游服务信息 标签的追加。你还可以自定义方法级别的标签&#xff0c;让日志的定位轻而易…

春秋云镜 CVE-2017-5638

春秋云镜 CVE-2017-5638 S2-045/S2-046 靶标介绍 2.3.32 之前的 Apache Struts 2 2.3.x 和 2.5.10.1 之前的 2.5.x 中的 Jakarta Multipart 解析器在文件上传尝试期间具有不正确的异常处理和错误消息生成&#xff0c;这允许远程攻击者通过精心制作的内容执行任意命令-Type、C…

Mysql 流程控制

简介 我们可以在存储过程和函数中实现比较复杂的业务逻辑&#xff0c;但是需要对应的流程控制语句来控制&#xff0c;就像Java中分支和循环语句一样&#xff0c;在MySQL中也提供了对应的语句&#xff0c;接下来就详细的介绍下。 1.分支结构 1.1 IF语句 IF 表达式1 THEN 操作1…

亚信科技AntDB数据库携“U8C+AntDB联合产品”亮相“2023全球商业创新大会”,开启生态合作新篇章

8月18-19日&#xff0c;近万人齐聚上海国家会展中心&#xff0c;带着对数字化、数智化趋势和热点的关注&#xff0c;以满腹热情投身到以“数据驱动 智能运营”为主题的“2023全球商业创新大会”&#xff0c;共商新技术条件下企业信息化出现的新课题、新挑战&#xff0c;共享数智…