当前位置: 首页 > web >正文

C语言---动态内存管理、柔性数组

一、malloc和free

1、变长数组

变长数组是指数组的大小可以通过变量来指定。

在c99以及之后的标准中:

#include<stdio.h>
int main()
{
int n=0;
scanf("%d",&n);
}

2、malloc和free

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

(1)若开辟成功,则返回一个指向开辟好空间的指针。

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

(3)返回类型是void* ,所以malloc函数并不知道开辟空间的类型,具体在使用时,使用者自己决定。

free是用来做动态内存的释放和回收的

释放ptr指向的空间,ptr中存放的是要释放的空间的起始地址。free函数用来释放动态开辟的内存。

注:(1)如果参数ptr指向的空间不是动态开辟的,那free函数的行为是为定义的

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

free(p);//释放
p=NULL;//置空指针
int main()
{int n = 5;int* arr = (int*)malloc(sizeof(int) * n);if (arr == NULL){perror("malloc failed");return 1;}for (int i = 0;i < n;i++){arr[i] = i + 1;}for (int i = 0;i < n;i++){printf("%d\n", arr[i]);}free(arr);arr = NULL;return 0;
}

3、calloc函数

功能:(1)为num个大小为size的元素开辟一块空间,并且把空间每个字节初始化为0.

(2)与函数malloc的区别在于calloc会在返回地址之前把申请的空间的每个字节初始化为全0。

int main()
{int n = 5;int* arr = (int*)calloc(5,sizeof(int) * n);if (arr == NULL){perror("calloc failed");return 1;}for (int i = 0;i < n;i++){arr[i] = i + 1;}for (int i = 0;i < n;i++){printf("%d\n", arr[i]);}free(arr);arr = NULL;return 0;
}

4、realloc函数----让动态内存管理更加灵活

realloc函数可以做到对动态开辟内存大小的调整。(ptr指向要调整的内存空间,ptr中存放的是要调整的内存空间的起始地址)

注:(常见动态申请访问)

1、对空指针的解引用操作(判定是否为空)

if(p==NULL)
{return 1;
}

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

3、对非动态开辟内存空间释放空间

4、对同一块多次释放

5、动态开辟内存忘记释放(内存泄漏)

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

二、柔性数组

结构体柔性数组(没指定大小,大小就是未知的,这就是柔性数组)

特点:(1)结构体中柔性数组成员前面必须至少一个其他成员

// 写法一
typedef struct {int num;int arr[]; // 柔性数组
} MyStruct1;// 写法二
typedef struct {int num;int arr[0]; // 柔性数组,部分编译器报错,建议用写法一
} MyStruct2;

(2)sizeof返回的这种结构大小不包括柔性数组的内存

(3)包含柔性数组成员的结构用malloc()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小。

MyStruct1 *p = (MyStruct1 *)malloc(sizeof(MyStruct1) + 5 * sizeof(int)); 
// 为柔性数组预留5个int空间

优势:1、方便内存管理:可以一次性分配结构体及柔性数组所需内存,释放时也只需要一次free操作,降低内存泄漏风险。

2、提高访问效率:内存连续,访问数组元素时无需多次寻址,减少访问次数,提升效率。

http://www.xdnf.cn/news/8425.html

相关文章:

  • unity控制相机围绕物体旋转移动
  • Maven打包SpringBoot项目,因包含SpringBootTest单元测试和Java预览版特性导致打包失败
  • 【leetcode】3356. 零数组变换②
  • 【uniapp】 iosApp开发xcode原生配置项(iOS平台Capabilities配置)
  • SFP与Unsloth:大模型微调技术全解析
  • 如何使用patch-package给npm包打补丁
  • 基于Java的话剧购票小程序【附源码】
  • 【Linux cmd】查找进程信息
  • Appium+python自动化(四)- 如何查看程序所占端口号和IP
  • Jmeter(三) - 测试计划(Test Plan)的元件
  • PostgreSQL 用户权限与安全管理
  • 量子传感器:开启微观世界的精准探测
  • Jetson系统烧录与环境配置全流程详解(含驱动、GCC、.Net设置)
  • 从JDK 8到JDK 17的主要变化
  • 《Medical SAM适配器:将分割一切模型适配于医学图像分割》|文献速递-深度学习医疗AI最新文献
  • Git企业级——进阶
  • NHANES指标推荐:MHR
  • leetcode:2469. 温度转换(python3解法,数学相关算法题)
  • docker swarm 启动容器报错日志查看方式
  • OpenHarmony 5.0中状态栏添加以太网状态栏图标以及功能实现
  • HTA8127内置升压的77W单体声D类音频功放
  • Vue3中reactive响应式使用注意事项
  • 【Java高阶面经:消息队列篇】24、Kafka消息顺序保障:单分区与多分区的性能优化
  • 贪心算法套路模板+详细适用场景+经典题目清单
  • 【PalladiumZ2 使用专栏 3 -- 信号值的获取与设置 及 memory dump 与 memory load】
  • MongoDB配置SSL
  • 【Fifty Project - D30】
  • 手抖人群饮食指南:科学膳食助力缓解震颤
  • MYSQL优化(1)
  • Python训练营打卡Day34