C语言动态内存相关函数

C语言动态内存管理函数有4个,分别为malloc,realloc,calloc和free。malloc函数分配一块堆内存;calloc是malloc的变种,功能相同,有细小的差别;realloc修改原内存块大小;free释放参数指针指向的内存块。下面分别介绍它们的函数原型、函数功能和一些特别的注意事项。


Function name


函数原型


函数功能


malloc


void * malloc ( size_t size );


向系统申请分配指定size个字节的内存空间。返回类型是 void* 类型。

void* 表示未确定类型的指针。void* 类型可以通过类型转换强制转换为任何其它类型的指针。


char *a = NULL;

//声明一个指向a的char*类型的指针

a = (char*)malloc(100*sizeof(char));

//使用malloc分配内存的首地址,然后赋值给a


realloc


void * realloc ( void * ptr, size_t size );


更改已经配置的内存空间的大小。可以减小也可以增大。


calloc


void * calloc ( size_t num, size_t size );


在内存的动态存储区中分配n个长度为size的连续空间,函数返回一个指向分配起始地址的指针;如果分配不成功,返回NULL。


char*str = NULL;

str = (char*)calloc(10,sizeof(char));

//calloc在动态分配完内存后,自动初始化该内存空间为零,而malloc不初始化,里边数据是随机的垃圾数据。


free


void free ( void * ptr );


释放之前由上面三个函数分配的内存块,以用于后面的再分配。需要注意的是,执行free后,指针ptr的值没有改变,仍然指向相同的(无效的)位置,而没有被置为NULL。

下面我们着重说明一下realloc。如果将分配的内存减少,realloc仅仅是改变索引的信息。

如果是将分配的内存扩大,则有以下情况:

1)如果当前内存段后面有需要的内存空间,则直接扩展这段内存空间,realloc()将返回原指针。

2)如果当前内存段后面的空闲字节不够,那么就使用堆中的第一个能够满足这一要求的内存块,将目前的数据复制到新的位置,并将原来的数据块释放掉,返回新的内存块位置。

3)如果申请失败,将返回NULL,此时,原来的指针仍然有效。

注意:如果调用成功,不管当前内存段后面的空闲空间是否满足要求,都会释放掉原来的指针,重新返回一个指针,虽然返回的指针有可能和原来的指针一样,即不能再次释放掉原来的指针。

看一下示例代码:

#include "stdafx.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

//函数原型:void *calloc(size_t n, size_t size);
//函数功能:在内存的动态存储区中分配n个长度为size的连续空间,
//          函数返回一个指向分配起始地址的指针;如果分配不成功,返回NULL。
void testCalloc()
{
    char*str = NULL;
    /*分配内存空间*/
    str = (char*)calloc(10,sizeof(char));
    /*将hello写入*/
    strcpy(str, "Hello");
    /*显示变量内容*/
    printf("String is %s\n",str);
    /*释放空间*/
    free(str);
    //与malloc的区别:
    //calloc在动态分配完内存后,自动初始化该内存空间为零,而malloc不初始化,里边数据是随机的垃圾数据。
    int i;
    int* pn = (int*)calloc(10, 2*sizeof(int));
    for(i = 0;i < 20;i++)
        printf("%d", pn[i]);
    printf("\n");
    free(pn);
}
void testRealloc()
{
    int input;
    int n;
    int *numbers1;
    int *numbers2;
    int *numbers3;
    numbers1=NULL;
    numbers3=NULL;

    if((numbers2=(int *)malloc(5*sizeof(int)))==NULL)//为numbers2在堆中分配内存空间
    {
        printf("malloc memory unsuccessful");
        exit(1);
    }

    printf("numbers2 addr: %8X\n",(int)numbers2);

    for(n=0;n<5;n++) //初始化
    {
        *(numbers2+n)=n;
        printf("numbers2‘s data: %d\n",*(numbers2+n));
    }

    printf("Enter new size: ");
    scanf("%d",&input);

    //重新分配内存空间,如果分配成功的话,就释放numbers2指针,
    //但是并没有将numbers2指针赋为NULL,也就是说释放掉的是系统分配的堆空间,
    //和该指针没有直接的关系,现在仍然可以用numbers2来访问这部分堆空间,但是
    //现在的堆空间已经不属于该进程的了。
    numbers3 = numbers2;
    numbers1=(int *)realloc(numbers2,(input+5)*sizeof(int));

    if(numbers1==NULL)
    {
        printf("Error (re)allocating memory");
        exit(1);
    }

    printf("numbers1 addr: %8X\n",(int)numbers1);
    printf("numbers3 addr: %8X\n",(int)numbers3);

    /*for(n=0;n<5;n++) //输出从numbers2拷贝来的数据
    {
        printf("the numbers1‘s data copy from numbers2: %d\n",*(numbers1+n));
    }*/

    for(n=0;n<input;n++)//新数据初始化
    {
        *(numbers2+5+n)=n+5;
        //printf("numbers1‘ new data: %d\n",*(numbers1+5+n));
    }

    free(numbers1);//释放numbers1,此处不需要释放numbers2,因为在realloc()时已经释放
    numbers1=NULL;
    //free(numbers2);//不能再次释放
    numbers2=NULL;
    //free(numbers3); 因为numbers3也指向numbers2指向的内存块,所以不能再次释放
    numbers3 = NULL;
}
int main(int argc, char* argv[], char* envp[])
{
    testCalloc();
    testRealloc();
    system("pause");
    return 0;
}

如果当前内存段后有足够的空间,realloc()返回原来的指针,但是,如果当前内存段后没有足够的空间,realloc()返回一个新的内存段的指针。

特别感谢下面两篇文章的作者,不明白的东西好像一下子就豁然开朗了,谢谢你们,所以我也希望把我的所学总结起来,能被更多需要的人看到。

参考文章:

http://www.cnblogs.com/younes/archive/2010/01/23/1654961.html

http://blog.csdn.net/hackerain/article/details/7954006

时间: 2024-10-15 00:36:38

C语言动态内存相关函数的相关文章

C语言动态内存管理

1-概述 动态存储管理的基本问题是:系统如何按请求分配内存,如何回收内存再利用.提出请求的用户可能是系统的一个作业,也可能是程序中的一个变量. 空闲块 未曾分配的地址连续的内存区称为"空闲块". 占用块 已分配给用户使用的地址连续的内存区称为"占用块". 系统刚刚启动时,整个内存可看做一个大的"空闲块",随着用户请求的进入,系统依次分配相应的内存. 在系统运行过程中,内存被分为两大部分:低地址区(若干占用块)和高地址区(空闲块). 经过一段时间后

C语言动态内存的申请和释放

什么是动态内存的申请和释放? 当程序运行到需要一个动态分配的变量时,必须向系统申请取得堆中的一块所需大小的存储空间,用于存储该变量.当不再使用该变量时,也就是它的生命结束时,要显式释放它所占用的存储空间,这样系统就能对该堆空间进行再次分配,做到重复使用有限的资源. 下面将介绍动态内存申请和释放的函数 1.malloc函数 在C语言中,使用malloc函数来申请内存.函数原型如下: #include<stdlib.h> void *malloc(size_t size); 参数size代表需要动

C语言动态内存

动态分配内存的概述 在数组一章中,介绍过数组的长度是预先定义好的,在整个程序中固定不变,但是在实际的编程中,往往会发生这种情况,即所需内存空间取决于实际输入的数据,而无法预先确定.为了解决上述问题,c语言提供了一些内存管理函数,这些内存管理函数可以按需要动态的分配内存空间,也可把不再使用的空间回收再次利用 静态分配.动态分配 静态分配 在程序编译或运行过程中,按事先规定大小分配内存空间的分配方式 int a[10] 必须事先知道所需空间的大小 分配在栈区或全局变量区,一般以数组的形式 按计划分配

C语言动态内存分配

1.new操作符和malloc函数 C++中的new操作符会隐式的考虑到分配的数据类型(与delete配合使用),表达式实例(以int类型为例) new int[4] malloc函数向系统申请一块指定大小的连续内存块(与free配合使用),需要一个参数来指定分配空间的字节大小,函数原型为 void* malloc(unsigned size) 表达式实例 malloc(4*sizeof(int)) malloc函数在内存的堆区中查找符合要求的内存块,若成功则做好标记表示已分配并返回指向该内存块

C语言学习笔记--动态内存分配

1. 动态内存分配的意义 (1)C 语言中的一切操作都是基于内存的. (2)变量和数组都是内存的别名. ①内存分配由编译器在编译期间决定 ②定义数组的时候必须指定数组长度 ③数组长度是在编译期就必须确定的 (3)但是程序运行的过程中,可能需要使用一些额外的内存空间 2. malloc 和 free 函数 (1)malloc 和 free 用于执行动态内存分配的释放 (2)malloc 所分配的是一块连续的内存 (3)malloc 以字节为单位,并且返回值不带任何的类型信息:void* mallo

C语言编程入门——动态内存分配

内存管理,是指软件运行时对计算机内存资源的分配和使用的技术.其最主要的目的是如何高效,快速的分配,并且在适当的时候释放和回收内存资源.内存管理是C语言编程中重要的组成部分,C语言中的内存需要手动分配,手动释放,一般遵循分配多少,释放多少,以免造成内存泄漏.内存管理是一项重要且复杂的事情,理解内存管理,对后面课程及项目的学习会有很大的作用. 之前创建变量,是系统自动分配的内存,放在栈内存中,销毁后被系统自动回收,手动分配的内存,放在堆内存中,需要手动释放.千万不要忘记销毁对象手动将内存释放并将指针

数据结构基础(1)--数组C语言实现--动态内存分配

数据结构基础(1)--数组C语言实现--动态内存分配 基本思想:数组是最常用的数据结构,在内存中连续存储,可以静态初始化(int a[2]={1,2}),可以动态初始化 malloc(). 难点就是数组在删除或者插入元素的时候,要移动元素的坐标不好确定.规律: 1.如果要在数组中第pos个位置插入一个元素(应该从后面开始移动) for( i=cnu;i>=pos;i--) pBase[i]=pBase[i-1]; 2.删除数组第pos位置的元素 for(i=pos+1;i<=cnu;i--)

【C语言天天练(九)】动态内存分配

引言:数组的元素存储于内存中连续的位置上.当一个数组被声明时.它所须要的内存在编译时就被分配. 可是,我们能够使用动态内存分配在执行时为它分配内存. 一块内存的生命周期能够分为四个阶段:分配.初始化.使用.释放. 内存的分配一般使用C函数库里的malloc函数(原型:void *malloc(size_t size)). 关于malloc函数应该注意一下几点: 1.malloc的參数就是须要分配的内存的字节数. 2.malloc所分配的是一块连续的内存. 3.分配成功.则返回指向分配内存起始地址

iOS开发之c语言基础Lesson-10 动态内存管理 上课笔记 与 试题练习

//函数声明 int max(int a, int b); char * getString(); int main(int argc, const char * argv[]) { //////////////////Lesson 10 动态内存管理 课堂笔记 和 练习 ///////复习上周所学 /////////常量和变量的区别: //常量中的内容不能被修改,只能访问: //变量中存储的数据时可以随时修改的. // //const 关键字的作用: 修饰变量不可改变.当做常量使用 //  c