【C/C++ 内存管理函数】C语言动态内存管理大揭秘:malloc、calloc、realloc与new的对比与差异

简介: 【C/C++ 内存管理函数】C语言动态内存管理大揭秘:malloc、calloc、realloc与new的对比与差异

C/C++内存管理专栏:C/C++ 内存管理专栏



C语言中内存分配方式

  1. 从静态存储区域分配
    -内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在.例如全局变量、static变量.
  2. 在栈上创建
    在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放.栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限.
  3. 从堆上分配,亦称动态内存分配
    -程序在运行的时候用malloc或new申请任意多少的内存,程序员自己负责在何时用free或delete释放内存.动态内存的生存期由用户决定,使用非常灵活,但问题也最多.

C语言内存申请相关函数

C语言跟内存申请相关的函数主要有 alloca、calloc、malloc、free、realloc等.

<1> alloca是向栈申请内存,因此无需释放.
<2> malloc分配的内存是位于堆中的,并且没有初始化内存的内容,因此基本上malloc之后,调用函数memset来初始化这部分的内存空间.
<3> calloc则将初始化这部分的内存,设置为0.
<4> realloc则对malloc申请的内存进行大小的调整.
<5> 申请的内存最终需要通过函数free来释放.


函数声明

接下来介绍正文了,三个函数的申明分别是:

void* malloc(unsigned size);
void* realloc(void* ptr, unsigned newsize);  
void* calloc(size_t numElements, size_t sizeOfElement);

三者都在stdlib.h函数库内,它们的返回值都是请求系统分配的地址,如果请求失败就返回NULL.
(1)函数malloc()
在内存的动态存储区中分配一块长度为size字节的连续区域,参数size为需要内存空间的长度,返回该区域的首地址.
如果由malloc()函数分配的内存空间原来没有被使用过,则其中的每一位可能都是0;反之, 如果这部分内存曾经被分配过,则其中可能遗留有各种各样的数据.
也就是说,使用malloc()函数的程序开始时(内存空间还没有被重新分配)能正常进行,但经过一段时间(内存空间还已经被重新分配)可能会出现问题。
(2)函数calloc()
与malloc相似,参数sizeOfElement为申请地址的单位元素长度,numElements为元素个数,即在内存中申请numElements*sizeOfElement字节大小的连续地址空间.
函数calloc() 会将所分配的内存空间中的每一位都初始化为零,也就是说,如果你是为字符类型或整数类型的元素分配内存,那么这些元素将保证会被初始化为0;
如果你是为指针类型的元素分配内存,那么这些元素通常会被初始化为空指针;
如果你为实型数据分配内存,则这些元素会被初始化为浮点型的零。
realloc可以对给定的指针所指的空间进行扩大或者缩小,无论是扩张或是缩小,原有内存的中内容将保持不变.当然,对于缩小,则被缩小的那一部分的内容会丢失。
realloc并不保证调整后的内存空间和原来的内存空间保持同一内存地址,相反,realloc返回的指针很可能指向一个新的地址。
(3)函数realloc()
给一个已经分配了地址的指针重新分配空间,参数ptr为原有的空间地址,newsize是重新申请的地址长度.
如果size较小,原来申请的动态内存后面还有空余内存,系统将直接在原内存空间后面扩容,并返回原动态空间基地址;
如果size较大,原来申请的空间后面没有足够大的空间扩容,系统将重新申请一块(20+size)*sizeof(int)的内存,并把原来空间的内容拷贝过去,原来空间free;如果size非常大,系统内存申请失败,返回NULL,原来的内存不会释放。
注意:如果扩容后的内存空间较原空间小,将会出现数据丢失,如果直接realloc(p, 0);相当于free(p).


malloc、calloc、realloc的区别

  • 初始化

    1.函数malloc申请的内存空间不会进行初始化,可能是随机值,
    2.函数calloc会将所分配的内存空间中的每一位都初始化为零.
    3.函数realloc申请的内存空间同样不会进行初始化,可能是随机值.
  • 功能
    1.malloc向系统申请分配指定size个字节的内存空间.返回类型是 void*类型.
    2.函数callocmalloc类似,但多一个参数,并不需要人为的计算空间的大小,适合申请指针数组空间.
    3.realloc功能更为丰富,可以实现内存分配和内存释放的功能,
    当扩大一块内存空间时,连续空间不够时,会使用堆上第一个有足够大小的自由块,现存的数据然后就被拷贝至新的位置,而老块则放回到堆上.重要的信息就是数据可能被移动。

C++ 动态分配内存:new

C/C++ 内存分配 new 操作符:剖析new操作符的实现机制和使用技巧

C/C++ 动态分配内存方式的原理和使用场景

C语言中的动态内存分配主要涉及到malloc、calloc、realloc这三个函数,而C++新增了new操作符。它们的底层原理和应用场景如下:

malloc(Memory ALLOCation):

原理:malloc是C语言中用于动态内存分配的函数。它向系统请求分配指定大小的内存空间,并返回指向这块内存的指针。malloc不会对分配的内存进行初始化,因此分配的内存可能包含垃圾数据。

应用场景:当你需要动态地分配内存空间时,可以使用malloc。这对于不确定数据大小的情况(如用户输入)特别有用。需要注意的是,分配的内存需要使用free函数释放,以防止内存泄漏。

calloc(Contiguous ALLOCation):

原理:calloc与malloc类似,也是用于动态内存分配的函数。与malloc的主要区别在于,calloc会初始化分配的内存空间,将其设置为0。因此,分配的内存不会包含垃圾数据。

应用场景:当你需要动态分配内存并确保其初始化为0时,可以使用calloc。与malloc相同,分配的内存需要使用free函数释放。

realloc(RE-ALLOCation):

原理:realloc函数用于调整之前分配的动态内存大小。如果需要扩大或缩小已分配内存的大小,realloc会尝试在原有内存的基础上调整大小,如果无法调整,则会分配一块新的内存,并将原有数据复制到新内存中。realloc不会对新分配的内存进行初始化。

应用场景:当你需要动态地调整内存大小时,可以使用realloc。例如,在处理可变长度数据结构(如动态数组)时,可能需要根据数据的增加或减少来调整内存。同样,需要使用free函数释放内存。

new(C++):

原理:new是C++中用于动态内存分配的操作符。与malloc和calloc相比,new具有更好的类型安全性,并能自动调用构造函数进行对象的初始化。同时,new操作符会根据请求的类型自动计算所需的内存大小。

应用场景:在C++中,可以使用new操作符分配动态内存,特别是在处理类和对象时。与malloc和calloc一样,分配的内存需要使用delete操作符释放。


总之,根据你的编程语言和需求,可以选择合适的动态内存分配方法。在C语言中,可以根据是否需要初始化内存来选择malloc或calloc,并使用realloc来调整内存大小。


结语

在我们的探索过程中,我们已经深入了解了C/C++的强大功能和广泛应用。然而,学习这些技术只是开始。真正的力量来自于你如何将它们融入到你的日常工作中,以提高效率和生产力。

心理学告诉我们,学习是一个持续且积极参与的过程。所以,我鼓励你不仅要阅读和理解这些命令,还要动手实践它们。尝试创建自己的命令,逐步掌握Shell编程,使其成为你日常工作的一部分。

同时,请记住分享是学习过程中非常重要的一环。如果你发现本博客对你有帮助,请不吝点赞并留下评论。

此外,我也欢迎你收藏本博客,并随时回来查阅。因为复习和反复实践也是巩固知识、提高技能的关键。

最后,请记住:每个人都可以通过持续学习和实践成为C/C++ 编程专家。我期待看到你在这个旅途中取得更大进步!

目录
相关文章
|
20天前
|
存储 缓存 NoSQL
工作 10 年!Redis 内存淘汰策略 LRU 和传统 LRU 差异,还傻傻分不清
小富带你深入解析Redis内存淘汰机制:LRU与LFU算法原理、实现方式及核心区别。揭秘Redis为何采用“近似LRU”,LFU如何解决频率老化问题,并结合实际场景教你如何选择合适策略,提升缓存命中率。
200 4
|
3月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
137 26
|
8月前
|
存储 程序员 编译器
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
4月前
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
70 1
|
11月前
|
程序员 C语言 开发者
pymalloc 和系统的 malloc 有什么区别
pymalloc 和系统的 malloc 有什么区别
|
7月前
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
10月前
|
存储 监控 算法
Java内存管理深度剖析:从垃圾收集到内存泄漏的全面指南####
本文深入探讨了Java虚拟机(JVM)中的内存管理机制,特别是垃圾收集(GC)的工作原理及其调优策略。不同于传统的摘要概述,本文将通过实际案例分析,揭示内存泄漏的根源与预防措施,为开发者提供实战中的优化建议,旨在帮助读者构建高效、稳定的Java应用。 ####
168 35
|
11月前
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
185 3
|
11月前
|
缓存 算法 Java
本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制
在现代软件开发中,性能优化至关重要。本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制。通过调整垃圾回收器参数、优化堆大小与布局、使用对象池和缓存技术,开发者可显著提升应用性能和稳定性。
179 6
|
11月前
|
程序员 C语言 开发者
pymalloc 和系统的 malloc 有什么区别?
pymalloc 和系统的 malloc 有什么区别?
150 2