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

【C语言】深入理解柔性数组:特点、使用与优势分析

C语言学习

柔性数组
友情链接:C语言专栏


文章目录

  • C语言学习
  • 前言:
  • 柔性数组
  • 一、柔性数组的特点
  • 二、柔性数组的使用
  • 三、柔性数组的优势
  • 总结
  • 附录
    • 上文链接
    • 专栏


前言:

在有结构体和动态内存分配的知识后,今天咱们来说说柔性数组吧!!!


柔性数组

C99 中,结构中的最后⼀个元素允许是未知大小的数组,这就叫做柔性数组成员。
例如:

struct S
{int i;int a[0];//柔性数组成员
};

有些编译器会报错⽆法编译可以改成:

struct S
{int i;int a[];//柔性数组成员
};

一、柔性数组的特点

  1. 结构中的柔性数组成员前面必须至少一个其他成员。
  2. sizeof 返回的这种结构大小不包括柔性数组的内存。
  3. 包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小。

示例:

struct S
{int i;int a[];//柔性数组成员
};
int main()
{printf("%d\n", sizeof(struct S));//输出的是4return 0;
}

输出:
在这里插入图片描述

二、柔性数组的使用

//代码1
#include<stdio.h>
#include<stdlib.h>
struct S
{int i;int a[];//柔性数组成员
};
int main()
{struct S* ps = (struct S*)malloc(sizeof(struct S) + 5 * sizeof(int));//动态开辟内存:4+20个字节//4个字节为i,20个字节为数组if (ps == NULL)//判断是否开辟成功{perror("malloc failed");exit(EXIT_FAILURE);  }//使用:ps->i = 100;for (int i = 0; i < 5; i++){ps->a[i] = i;}//增加空间reallocstruct S* tmp = (struct S*)realloc(ps, sizeof(struct S) + 10 * sizeof(int)); //给数组增加20个字节(5个元素)if (tmp == NULL)//判断是否开辟成功{perror("malloc failed");exit(EXIT_FAILURE);}ps = tmp;//继续由ps管理这块内存;tmp = NULL;//使用//……//释放free(ps);ps = NULL;return 0;
}

三、柔性数组的优势

上述的结构也可以设计为别的结构,也能完成同样的效果。

//代码2
#include<stdio.h>
#include<stdlib.h>
struct S
{int i;int* a;//通过指针指向一块空间,实现与柔性数组相同的效果
};
int main()
{struct S* ps = (struct S*)malloc(sizeof(struct S));//动态开辟内存一个struct S大小的空间if (ps == NULL)//判断是否开辟成功{perror("malloc failed");exit(EXIT_FAILURE);}ps->i = 100;ps->a = (int*)malloc(5 * sizeof(int)); //动态开辟内存5个int大小的空间,并将起始地址给aif (ps->a == NULL)//判断是否开辟成功{perror("malloc failed");exit(EXIT_FAILURE);}//使用for (int i = 0; i < 5; i++){ps->a[i] = i;}//增加空间reallocint* tmp = (int*)realloc(ps->a, sizeof(10 * sizeof(int)));//给ps->a指向的空间增加20个字节(5个元素)if (tmp == NULL)//判断是否开辟成功{perror("malloc failed");exit(EXIT_FAILURE);}ps->a  = tmp;tmp = NULL;//使用//……//释放//注意先后顺序free(ps->a );ps->a  = NULL;free(ps);ps = NULL;return 0;
}

上述代码1和代码2可以完成同样的功能,但是代码1的实现有两个好处:
第一个好处是:方便内存释放

咱们上面发现,代码2我们会释放两次内存。有如果我们的代码是在⼀个给别⼈用的函数中,你在里面做了二次内存分配,并把整个结构体返回给用户。用户调用free可以释放结构体,但是用户并不知道这个结构体内的成员也需要free,所以你不能指望用户来发现这个事。所以,如果我们把结构体的内存以及其成员要的内存⼀次性分配好了,并返回给用户⼀个结构体指针,用户做⼀次free就可以把所有的内存也给释放掉。

第二个好处是:这样有利于访问速度

连续的内存有益于提高访问速度,也有益于减少内存碎片。图示:
在这里插入图片描述


总结

柔性数组是C99标准引入的重要特性,它允许在结构体的末尾定义一个长度可变的数组。通过结合动态内存管理,柔性数组能够高效地处理不确定大小的数据集合,是C程序员进阶必备技能之一。

附录

上文链接

《动态内存分配避坑指南:六大易错点解析与经典笔试题实战》

专栏

C语言专栏

http://www.dtcms.com/a/285566.html

相关文章:

  • Cursor替代,公测期间免费使用Claude4
  • 首个直播流扩散(LSD)AI模型:MirageLSD,它可以实时把任意视频流转换成你的自定义服装风格——虚拟换装新体验
  • mpiigaze的安装过程一
  • 【后端】.NET Core API框架搭建(10) --配置163邮件发送服务
  • 【锂电池剩余寿命预测】TCN时间卷积神经网络锂电池剩余寿命预测(Pytorch完整源码和数据)
  • C#之线程Thread
  • ARCS系统机器视觉实战(直播回放)
  • Huber Loss(胡贝损失)详解:稳健回归的秘密武器 + Python实现
  • Unity 堆栈分析实战指南 C#
  • Copula 回归与结构方程模型:R 语言构建多变量因果关系网络
  • 机器视觉的布料丝印应用
  • React条件渲染
  • 用 React-Three-Fiber 实现雪花下落与堆积效果:从零开始的 3D 雪景模拟
  • jvm分析篇---1、先认识下dump文件
  • Linux系统安装Docker及部署Node.js 20.15.0(含pnpm、pm2)完整指南
  • Docker部署前后端分离项目——多项目共享环境部署
  • GEV/POT/Markov/点过程/贝叶斯极值全解析;基于R语言的极值统计学
  • Camera相机人脸识别系列专题分析之十七:人脸特征检测FFD算法之libhci_face_camera_api.so 296点位人脸识别检测流程详解
  • vue2 面试题及详细答案150道(81 - 90)
  • 2025阿里云黑洞自救指南:从分钟级恢复到长效免疫的实战方案
  • RPG59.玩家拾取物品三:可拾取物品的提示UI
  • Excel批量生成SQL语句 Excel批量生成SQL脚本 Excel拼接sql
  • Android studio和gradle升级后的一些错误
  • YOLO 目标检测的改进方法
  • FastExcel:革新Java生态的高性能Excel处理引擎
  • [2025CVPR-目标检测方向]FSHNet:一种用于3D物体检测的全稀疏混合网络。
  • 如何为“地方升学导向型”语校建模?Prompt 框架下的宇都宫日建工科专门学校解析(7 / 500)
  • 20250718-6-Kubernetes 调度-Pod对象:环境变量,初始容器,静态_笔记
  • Python练习(6)Python面向对象编程三大特性:封装、继承与多态的15道实战练习题(含答案与深度解析)
  • 一文讲透HTML语义化标签