realloc(void *__ptr, size_t __size):更改已经配置的内存空间,即更改由malloc()函数分配的内存空间的大小。如果将分配的内存减少,realloc仅仅是改变索引的信息。
1.如果realloc()函数的第一个参数是NULL,那么系统就会分配第二个参数指定的新内存,并返回新内存单元的首字节的指针。此时,realloc()函数类似于malloc()函数。如果第一个参数不是NULL,但不指向以前分配的内存,或者指向已经free的内存单元,那么结果就是不确定的。
2.realloc()函数保存了第一个参数对应的内存单元的内容,且保存的量是新旧内存单元里较小的那一个。如果新内存单元大于旧内存单元,那么新增的内存就不会被初始化,而是包含垃圾值。
3.使用内存分配函数时,应该尽量避免多次分配小的内存单元,因为分配堆上的内存有一些系统开销,所以分配许多小的内存单元比分配几个大的内存单元的系统开销大。
如果是将分配的内存扩大,则有以下情况:1)如果当前内存段后面有需要的内存空间,则直接扩展这段内存空间,realloc()将返回原指针。
2)如果当前内存段后面的空闲字节不够,那么就使用堆中的第一个能够满足这一要求的内存块,将目前的数据复制到新的位置,并将原来的数据块释放掉,返回新的内存块位置。
3)如果申请失败,将返回NULL,此时,原来的指针仍然有效。
注意:如果调用成功,不管当前内存段后面的空闲空间是否满足要求,都会释放掉原来的指针,重新返回一个指针,虽然返回的指针有可能和原来的指针一样,即不能再次释放掉原来的指针。
看一下示例代码
如果当前内存段后有足够的空间,realloc()返回原来的指针:
如果当前内存段后没有足够的空间,realloc()返回一个新的内存段的指针:
calloc()calloc()函数是malloc的简单包装。它的主要优点是把动态分配的内存清零。
void *calloc(size_t nmemb, size_t size);
1. calloc()函数相比malloc()函数的优点是,将所分配的内存初始化为0;
2. 范例中calloc()函数的参数顺序是先size后个数,与函数定义中先个数后size的格式不同,编译使用都没有问题,但是正确的写法可读性强一些,而且养成规范的习惯可以减少很多不必要的出错机会,和正常的说话顺序也很接近,比如我们都说“我要5个int”,而不说“我要int 5个”;
定义中的参数顺序比较规范,应按照定义中给出的顺序写代码。
用经验的程序员更喜欢使用calloc(),因为这样的话新分配内存的内容就不会有什么问题,调用calloc()肯定会清0,并且可以避免调用memset().
memset
功 能: 将s所指向的某一块内存中的每个字节的内容全部设置为ch指定的ASCII值,
块的大小由第三个参数指定,这个函数通常为新申请的内存做初始化工作
用 法: void *memset(void *s, char ch, unsigned n);
程序例:
#include <string.h>
#include <stdio.h>
#include <mem.h>
int main(void)
{
char buffer[] = "Hello world\n";
printf("Buffer before memset: %s\n", buffer);
memset(buffer, '*', strlen(buffer) );
printf("Buffer after memset: %s\n", buffer);
return 0;
}
输出结果:
Buffer before memset: Hello world
Buffer after memset: ***********
编译平台:
Microsoft Visual C++ 6.0