• 动态内存管理


    目录

    内存中的栈区和堆区

    malloc

    free

    calloc

    realloc

    内存中的栈区和堆区

    我们知道php的底层是C (任何语言其实都可以分为大同小异的几块)

    而C语言的内存模型分为5个区:栈区、堆区、静态区、常量区、代码区。每个区存储的内容如下:

    1、栈区:存放函数的参数值、局部变量等,由编译器自动分配和释放,通常在函数执行完后就释放了,其操作方式类似于数据结构中的栈。栈内存分配运算内置于CPU的指令集,效率很高,但是分配的内存量有限,比如iOS中栈区的大小是2M。

    2、堆区:就是通过new、malloc、realloc分配的内存块,编译器不会负责它们的释放工作,需要用程序区释放。分配方式类似于数据结构中的链表。“内存泄漏”通常说的就是堆区。

    3、静态区:全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域,未初始化的全局变量和未初始化的静态变量在相邻的另一块区域。程序结束后,由系统释放。

    4、常量区:常量存储在这里,不允许修改。

    5、代码区:顾名思义,存放代码。

    分布图:

     

     

     

    malloc

    c语言提供了一个动态内存开辟的函数:

    void* malloc(sizt_t size);

    这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。 

    1、如果开辟成功,则返回一个指向开辟好空间的指针

    2、如果开辟失败,则返回一个指向NULL的指针,因此malloc的返回值一定要做检查。例如:

    1. int* p = (int*)malloc(10*sizeof(int));
    2. int* ptr = p;
    3. if (p == NULL)
    4. {
    5. printf("%s\n", strerror(errno));
    6. return 1;
    7. }
    8. //上面第一行代码,因为malloc返回值是void*类型,(int*)会将malloc返回值
    9. //强制类型转换为int*类型
    10. //sizeof的单位是字节;
    11. //malloc()括号里面填写的是字节大小,返回值为int*类型方便后面使用这块空间时解引用 p时,
    12. //*(p+1)跳过的是一个整型大小,方便数据的存储

    3、返回值的类型是void,所以malloc函数并不知道开辟空间的类型,具体在使用的时候由自己决定

    4、如果参数size为0,malloc的行为标准是未定义的,取决于编译器。 

    malloc函数申请动态内存空间的代码:

    free

    专门是用来做动态内存的释放和回收的,函数原型如下:

    void free(void * ptr);

    free函数用来释放动态开辟的内存。

    1、如果参数ptr指向的空间不是动态开辟的,那free函数的行为是未定义的。

    2、如果参数ptr是NULL指针,则函数什么事也不做。

    malloc和free都声明在stdlib.h头文件中

     举个例子:

    1. int main()
    2. {
    3. //int arr[10];//向内存申请了40个字节
    4. int* ptr = (int*)malloc(10*sizeof(int));
    5. int* p = ptr;
    6. if (p == NULL)
    7. {
    8. printf("%s\n", strerror(errno));
    9. return 1;
    10. }
    11. //使用
    12. int i = 0;
    13. for (i = 0; i < 10; i++)
    14. {
    15. *p = i;
    16. p++;
    17. }
    18. //释放
    19. free(ptr);
    20. p = NULL;
    21. ptr = NULL;
    22. return 0;
    23. }

    free函数释放空间是从ptr指向的地址后面的地址空间:在使用的时候指针ptr往后移动,但是释放空间的时候需要释放ptr后面的空间而不是p之后的

    所以一般用free()的时候传入的是申请空间的时候的旧地址 

                  malloc      free    成对出现

                  calloc       free     成对出现

    这里需要注意的是:假如申请一块内存空间,则需要在不使用的时候及时释放掉,否则会造成内存泄漏(你申请了一块内存,自己不用别人也用不着)

    还会一直占用内存,直到程序运行结束

    例如:

    1. while(1)
    2. {
    3. malloc(40);
    4. }

     假如在一个操作系统里面,这里需要申请内存,那里也需要申请内存空间,那么就会占用大量的内存空间。

    malloc函数申请动态内存空间,和free函数释放的代码:

    1. #include
    2. #include
    3. #include
    4. INT_MAX
    5. int main()
    6. {
    7. //int arr[10];//向内存申请了40个字节
    8. int* p = (int*)malloc(10*sizeof(int));
    9. int* ptr = p;
    10. if (p == NULL)
    11. {
    12. printf("%s\n", strerror(errno));
    13. return 1;
    14. }
    15. //使用
    16. int i = 0;
    17. for (i = 0; i < 10; i++)
    18. {
    19. *ptr = i;
    20. ptr++;
    21. }
    22. //释放
    23. free(p);
    24. p = NULL;
    25. ptr = NULL;
    26. return 0;
    27. }

    calloc

    calloc也用来做动态内存分配,原型如下:

    1. void*calloc(size_t num,size_t size);
    2. //这里siez_t指的是无符号整型
    3. //num指的是个数
    4. //size 表示大小
    5. //总的来说就是申请num个大小为size字节的空间

    举个例子: 

    int *p = (int*)calloc(10,sizeof(int));

     它的返回值类型也是void*,可以通过强制类型转换(int*)转换为整型指针,int大小为4个字节

    realloc

    realloc函数的出现让动态内存管理更加灵活。

    有时我们会发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的使用内存,我们一定会堆内存的大小做灵活的调整。那realloc就可以做到对动态开辟内存大小的调整。函数原型如下:

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

    1、ptr是要调整的内存地址

    2、size调整之后的新大小,(这里的size一般要比原来的大,不然可能会造成数据丢失)

    3、这个函数调整原地址空间大小的基础上,还会将原来内存中的数据移动到新的空间。例如:

     

            当原来的地址空间的后面无法开辟需要的大小的空间的时候,会异地开辟一个新的地址空间。首先会把原来的数据复制到新的地址空间,然后会自动释放掉原来的空间,最后返回新的地址空间的地址。 

    realloc动态申请空间的代码:

    1. int main()
    2. {
    3. int* p = (int*)malloc(40);
    4. if (p == NULL)
    5. return 1;
    6. //使用
    7. int i = 0;
    8. for (i = 0; i < 10; i++)
    9. {
    10. *(p + i) = i;
    11. }
    12. //
    13. for (i = 0; i < 10; i++)
    14. {
    15. printf("%d ", *(p + i));
    16. }
    17. //增加空间
    18. int* ptr = (int*)realloc(p, 80);
    19. //当realloc开辟失败的是,返回的是NULL
    20. //....
    21. if (ptr != NULL)
    22. {
    23. p = ptr;
    24. ptr = NULL;
    25. }
    26. for (i = 10; i < 20; i++)
    27. {
    28. *(p + i) = i;
    29. }
    30. //释放
    31. free(p);
    32. p = NULL;
    33. return 0;
    34. }

  • 相关阅读:
    Flask-[实现websocket]-(2): flask-socketio文档学习
    概念解析 | 神经网络中的位置编码(Positional Encoding)
    一文搞懂ES6的Map
    Spark Dataset 输出成csv文件
    九齐NY8BE62D/NY8B062D中硬件PWM的用法
    海康威视监控相机的SDK与opencv调用(非工业相机)
    BiLSTM(双向LSTM)实现股票时间序列预测(TensorFlow2版)
    Android虚拟机线程启动过程解析, 获取Java线程真实线程Id的方式
    python文件操作之xml转txt
    ros2原来本是一个通信协议
  • 原文地址:https://blog.csdn.net/weixin_53269843/article/details/127827871