• 【C语言】-动态内存管理



    前言

    为什么存在动态内存分配?

    我们学习过变量,学习过数组。
    但是我们知道,我们声明变量时,例如int型变量,它其实是在栈上开辟了4个字节的空间,当我们声明长度为10,类型为int型的数组时,它其实是在栈上开辟了40个字节的空间,一旦我们实现开辟好,那么就面临一个问题,这个空间多了还是少了?用不用的完?用不完会造成空间浪费,而少了会导致内存溢出。
    这时候,动态内存分配就起到了作用。


    1、动态内存管理函数的介绍

    1.1、malloc和free

    C语言给我们提供了动态开辟内存的函数:malloc

    malloc函数的基本形式:

    void* malloc(size_t size);

    malloc函数的作用:

    向内存申请一块size个字节大小的空间。

    函数的返回值:

    • 如果开辟成功,则返回开辟好的空间的地址。
    • 如果开辟失败,则返回空指针NULL,这也就是为什么malloc空间之后一定要做指针是否为NULL的判断的原因。

    malloc函数使用时需要注意的事项:

    • malloc函数返回的是void*型的指针,所以需要我们在使用时,自己将指针转换为我们需要的指针类型。
    • size的大小为0时,malloc的行为是未定义的,具体取决于编译器。这个行为一般没有什么意义。
    • 返回的指针用p接收了之后,就不要随意改动p的位置,因为当我们改动了p的位置之后,我们就找不到那片空间的起始位置了,而当我们free的时候又是以p为起始位置开始释放那片空间。最好的解决办法就是先将p赋给p1,让p1去变动。
    • 把p指针free之后,要记得把p指针置为NULL,因为此时p指向的那片空间已经还给操作系统了,此时p指针属于野指针。

    C语言给我们提供了用来释放动态开辟的内存的工具:free
    free函数使用的注意事项:

    • 如果参数指针指向的空间不是动态开辟的,那么free函数的行为是未定义的。
    • 如果参数指针是NULL指针,则函数就什么都不做。

    ps:malloc和free都声明在stdlib.h头文件中
    使用案例:

    #include 
    int main()
    {
     //代码1
     int num = 0;
     scanf("%d", &num);
     int arr[num] = {0};
     //代码2
     int* ptr = NULL;
     ptr = (int*)malloc(num*sizeof(int));
     if(NULL != ptr)//判断ptr指针是否为空
     {
     int i = 0;
     for(i=0; i<num; i++)
     {
     *(ptr+i) = 0}
     }
     free(ptr);//释放ptr所指向的动态内存
     ptr = NULL;//是否有必要?
     return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    1.2、calloc

    C语言提供了一个函数叫calloc,calloc也可以用来动态内存分配。
    calloc基本样式如下:

    void* calloc(size_t num,size_t size);

    calloc的基本功能:

    为num个大小为size的元素开辟一块内存空间,并且把空间的每一个字节初始化为0。

    ps:calloc和malloc的区别只在于calloc会在返回地址之前把申请的内存空间每个字节全都初始化为0。

    1.3、realloc

    同样的,realloc也是C语言给我们提供的函数
    realloc函数的基本形式:

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

    realloc函数的功能:

    调整申请的空间的大小。ptr是要调整的内存地址,size是调整之后的大小。返回调整之后的内存的起始地址。

    realloc函数的注意事项:

    涉及到调整大小,那么就会出现一个问题:如果我要在原有空间的基础上调大空间,但是后面没有足够大的空间了怎么办?

    ![在这里插入图片描述](https://img-blog.csdnimg.cn/c758d49e30854635a768c86601e3f903.png
    在这里插入图片描述

    2、常见动态内存的错误

    2.1、对NULL指针的解引用

    void test()
    {
     int *p = (int *)malloc(INT_MAX/4);
     *p = 20;//如果p的值是NULL,就会有问题
     free(p);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    这就犯了我们前面提到的错误,malloc之后要判断返回的指针是否为NULL指针

    2.2、对动态开辟空间的越界访问

    void test()
    {
     int i = 0;
     int *p = (int *)malloc(10*sizeof(int));
     if(NULL == p)
     {
     exit(EXIT_FAILURE);
     }
     for(i=0; i<=10; i++)
     {
     *(p+i) = i;//当i是10的时候越界访问
     }
     free(p);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    当我们用指针接收了开辟的内存的地址后,我们对这个开辟的内存地址外的地址进行访问。 因为外面那片空间不属于我们,导致了对开辟空间的越界访问。

    2.3、对非动态开辟的内存空间使用了free释放

    void test()
    {
     int a = 10;
     int *p = &a;
     free(p);//ok?
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    我们前面提到,free只能对动态开辟的内存空间使用。这里对非动态开辟的内存进行了释放是错误的。

    2.4、使用free释放了一块动态开辟的内存的一部分

    void test()
    {
     int *p = (int *)malloc(100);
     p++;
     free(p);//p不再指向动态内存的起始位置
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    我们前面提到,我们尽量不要对p指针进行修改,修改后p指针就没有指向那块空间的起始地址了,这时候free(p)的话就只释放掉了动态开辟的内存的后面那部分。我们在使用那块空间时,应该声明一个变量p1,把p指向的地址赋给p1。

    3.5、对同一块动态内存的多次释放

    void test()
    {
     int *p = (int *)malloc(100);
     free(p);
     free(p);//重复释放
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    3.6、动态开辟的内存未释放(内存泄漏

    void test()
    {
     int *p = (int *)malloc(100);
     if(NULL != p)
     {
     *p = 20;
     }
    }
    int main()
    {
     test();
     while(1);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    忘记释放会导致,每次调用这个函数都会在内存开辟一块空间,这样空间越来越少,最终导致死机,关机重启之后程序又能继续运行。

  • 相关阅读:
    字节码进阶之JVM Attach API详解
    宠物喂食器方案智能开发设计
    安装mysql
    深度学习_2_数据处理
    一段代码让你了解Java中的抽象
    卷积神经网络 图像处理,图像识别卷积神经网络
    蓝桥杯官网填空题(梅森素数)
    随笔记录-看nacos源码
    一文搞懂什么是 PostCSS
    EA&UML日拱一卒 用例包含关系
  • 原文地址:https://blog.csdn.net/flin666/article/details/127691867