(转):alloca/malloc/calloc/realloc/sbrk/new/delete

C语言跟内存分配方式

  (1) 从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在。例如全局变量,static变量。

  (2) 在栈上创建。在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。

  (3)从堆上分配,亦称动态内存分配。程序在运行的时候用malloc或new申请任意多少的内存,程序员自己负责在何时用free或delete释放内存。动态内存的生存期由我们决定,使用非常灵活,但问题也最多

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

  其中alloca是向栈申请内存,因此无需释放. malloc分配的内存是位于堆中的,并且没有初始化内存的内容,因此基本上malloc之后,调用函数memset来初始化这部分的内存空间.

  calloc则将初始化这部分的内存,设置为0. 而realloc则对malloc申请的内存进行大小的调整.申请的内存最终需要通过函数free来释放. 而sbrk则是增加数据段的大小;

  malloc/calloc/free基本上都是C函数库实现 的,跟OS无关.C函数库内部通过一定的结构来保存当前有多少可用内存.如果程序 malloc的大小超出了库里所留存的空间,那么将首先调用brk系统调用来增加可用空间,然后再分配空间.free时,释放的内存并不立即返回给os, 而是保留在内部结构中. 可以打个比方: brk类似于批发,一次性的向OS申请大的内存,而malloc等函数则类似于零售,满足程序运行时的要求.这套机制类似于缓冲.

  使用这套机制的原因: 系统调用不能支持任意大小的内存分配(有的系统调用只支持固定大小以及其倍数的内存申请,这样的话,对于小内存的分配会造成浪费; 系统调用申请内存代价昂贵,涉及到用户态和核心态的转换. 函数malloc()和calloc()都可以用来分配动态内存空间,但两者稍有区别。

  malloc()函数有一个参数,即要分配的内存空间的大小:

  Void *malloc(size_t size);

  calloc()函数有两个参数,分别为元素的数目和每个元素的大小,这两个参数的乘积就是要分配的内存空间的大小:

  void*calloc(size_t numElements,size_t sizeOfElement);

  如果调用成功,函数malloc()和calloc()都将返回所分配的内存空间的首地址。

  malloc() 函数和calloc()函数的主要区别是前者不能初始化所分配的内存空间,而后者能。如果由malloc()函数分配的内存空间原来没有被使用过,则其中 的每一位可能都是0;反之,如果这部分内存空间曾经被分配、释放和重新分配,则其中可能遗留各种各样的数据。也就是说,使用malloc()函数的程序开 始时(内存空间还没有被重新分配)能正常运行,但经过一段时间后(内存空间已被重新分配)可能会出现问题。

  calloc() 函数会将所分配的内存空间中的每一位都初始化为零,也就是说,如果你是为字符类型或整数类型的元素分配内存,那么这些元素将保证会被初始化为零;如果你是 为指针类型的元素分配内存,那么这些元素通常(但无法保证)会被初始化为空指针;如果你是为实数类型的元素分配内存,那么这些元素可能(只在某些计算机 中)会被初始化为浮点型的零。

  malloc() 函数和calloc()函数的另一点区别是calloc()函数会返回一个由某种对象组成的数组,但malloc()函数只返回一个对象。为了明确是为一 个数组分配内存空间,有些程序员会选用calloc()函数。但是,除了是否初始化所分配的内存空间这一点之外,绝大多数程序员认为以下两种函数调用方式 没有区别:

  calloc(numElements ,sizeOfElement);

  malloc(numElements *sizeOfElement) ;

  需要解释的一点是,理论上(按照ANSIC标准)指针的算术运 算只能在一个指定的数组中进行,但是在实践中,即使C编译程序或翻译器遵循这种规定,许多C 程序还是冲破了这种限制。因此,尽管malloc()函数并不能返回一个数组,它所分配的内存空间仍然能供一个数组使用(对realloc()函数来说同 样如此,尽管它也不能返回一个数组)。

  总之,当你在calloc()函数和malloc()函数之间作选择时,你只需考虑是否要初始化所分配的内存空间,而不用考虑函数是否能返回一个数组。

  当程序运行过程中malloc了,但是没有free的话,会造 成内存泄漏.一部分的内存没有被使用,但是由于没有free,因此系统认为这部分内存还在使用,造成不断的向系统申请内存,使得系统可用内存不断减少.但 是,内存泄漏仅仅指程序在运行时,程序退出时,OS将回收所有的资源.因此,适当的重起一下程序,有时候还是有点作用.

realloc可以对给定的指针所指的空间进行扩大或者缩小,无论是扩张或是缩小,原有内存的中内容将保持不变。当然,对于缩小,则被缩小的那一部分的内容会丢失。

realloc 并不保证调整后的内存空间和原来的内存空间保持同一内存地址。相反,realloc 返回的指针很可能指向一个新的地址。所以在代码中,我们必须将realloc返回的值,重新赋值给 p 

new 相当于调用malloc为对象分配内存并调用对象的构造函数,delete相当于调用free释放内存并调用对象的析构函数,new产生的内存被认为是该 对象类型的,可以直接使用,而malloc只返回void*,若要变为对象类型使用还需进行强制转换;另外new/delete是运算符,而malloc 和free是系统函数,在c++中,由于语言内置类型无构造和析构函数,所以new/delete与malloc/free相当,因而new /delete可以完全替代malloc/free的功能。

转载于:https://www.cnblogs.com/yysblog/archive/2012/03/31/2427442.html

### 回答1: 这些函数都是在C语言中用来动态分配内存的函数,它们的特点如下: 1. alloca函数:是一个非标准的函数,它分配内存的方式与malloc不同,它是在栈上分配内存,所以内存的生命周期是与函数的生命周期相同的。使用alloca分配的内存不需要手动释放,在函数返回时会自动释放。 2. malloc函数:是标准库函数,用于在堆上分配指定大小的内存。使用完成后需要手动释放,否则会导致内存泄漏。 3. calloc函数:也是标准库函数,用于在堆上分配指定大小的内存,并且将其初始化为0。使用完成后需要手动释放,否则会导致内存泄漏。 4. realloc函数:是标准库函数,用于重新分配已经分配的内存大小。如果之前分配的内存不足以满足新的内存需求,realloc会重新分配一块更大的内存,并将原有内存中的内容复制到新分配的内存中。使用完成后需要手动释放,否则会导致内存泄漏。 ### 回答2: allocamalloccalloc和relloca是在C语言中用于动态内存分配的函数。 alloca是C语言中的一个可变分配内存函数。它是在栈上分配内存的,用于动态地创建一个连续的内存块。它的特点是不需要手动释放内存,当函数返回时自动释放。但是,由于在栈上分配内存,它的内存分配和释放速度比较快,但分配的内存大小有限。 malloc是C语言中常用的动态内存分配函数。它在堆上分配内存,可以动态地创建任意大小的内存块。它的特点是需要手动释放内存,通过调用free函数来释放内存。由于在堆上分配内存,其分配和释放速度比较慢,但可以分配较大的内存空间。 calloc是C语言中另一个动态内存分配函数。它在堆上分配内存,并将分配的内存块清零。它的特点是需要手动释放内存,通过调用free函数来释放内存。与malloc相比,calloc适用于创建数组或矩阵等需要初始化的数据结构,因为可以保证分配的内存块全部为0。 relloca是一种特殊的动态内存分配函数,它在堆上重新分配内存。它的特点是可以增大或减小已分配内存块的大小。如果需要调整已分配内存块的大小,可以使用relloc函数,它会分配一个新的内存块,并将原有内存块的内容拷贝到新的内存块,然后释放原内存块。需要注意的是,使用realloc时需要特别小心,因为当无法满足大小调整时,它会返回NULL。 这些函数的特点不同,选择使用哪个函数取决于具体的需求和场景。 ### 回答3: allocamalloccallocrealloc都是用于在内存中分配空间的函数: 1. alloca函数: alloca函数用于在栈上分配空间,该空间在函数调用结束后自动释放,不需要手动释放。它的特点如下: - alloca是计算栈帧大小的一部分,速度较快。 - 分配的内存空间是在栈上,所以不需要显式释放内存。 - 分配的内存空间是连续的,在函数调用结束后会被自动释放。 - 分配的内存空间大小限制较小。 2. malloc函数: malloc函数用于在堆上分配指定大小的内存空间,并返回一个指向该内存空间的指针。它的特点如下: - 分配的内存空间是在堆上,需要手动释放。 - 分配的内存空间大小不受限制,可以根据需要动态调整。 - 由于需要维护堆的空闲内存链表,所以执行速度相对较慢。 3. calloc函数: calloc函数用于在堆上分配指定数量和大小的连续内存空间,并将其初始化为零。它的特点如下: - 分配的内存空间是在堆上,需要手动释放。 - 分配的内存空间大小不受限制,可以根据需要动态调整。 - 分配的内存空间会被初始化为零。 - 适用于需要分配并初始化一块连续内存空间的情况。 4. realloc函数: realloc函数用于重新分配已有内存块的大小,可以扩大或缩小。它的特点如下: - 分配的内存空间是在堆上,需要手动释放。 - 可以扩大或缩小已有内存块的大小。 - 如果重新分配的大小比原有大小大,会尝试在原有内存块后续位置进行扩展;如果重新分配的大小比原有大小小,会截断内存块。 - 可能导致内存拷贝或内存移动,效率较低。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值