柔性数组(C语言)

柔性数组是 C99 标准中引入的一种特殊的数组,它可以用作结构体的最后一个成员,以便动态分配内存时灵活管理数组的大小。


1. 柔性数组的定义

柔性数组是结构体的最后一个成员,声明时不需要指定大小。例如:

  1. struct FlexibleArray {
        int length;
        char data[];  // 柔性数组成员,大小未指定
    };
    

    这里的 data[] 是一个柔性数组成员。它没有固定的大小,可以在运行时根据需要动态分配。

    特点:

  2. 必须是结构体的最后一个成员
  3. 不能用于联合体(union)
  4. 声明时不能指定数组大小(例如 char data[0]char data[10] 都不允许)。
  5. 柔性数组不占用结构体的内存空间(未分配数据之前),其大小为零。

 

2. 使用柔性数组的场景

柔性数组通常用于以下场景:

  1. 动态大小的数组:在运行时根据需求分配不同大小的数组。
  2. 内存优化:避免在结构体中声明一个固定大小的数组,从而减少不必要的内存浪费。
  3. 实现复杂数据结构:如动态表、消息帧等。
  1. 3. 柔性数组的内存布局

    柔性数组成员不会占用结构体的固定大小,而是通过动态分配内存来扩展。

    示例:

    #include <stdio.h>
    #include <stdlib.h>
    
    struct FlexibleArray {
        int length;    // 数组长度
        char data[];   // 柔性数组成员
    };
    
    int main() {
        // 分配结构体和柔性数组的内存
        int n = 10;
        struct FlexibleArray *fa =(FlexibleArray *) malloc(sizeof(struct FlexibleArray) + n * sizeof(char));
    
        if (fa == NULL) {
            printf("内存分配失败!\n");
            return -1;
        }
    
        // 设置数组长度
        fa->length = n;
    
        // 初始化柔性数组
        for (int i = 0; i < n; i++) {
            fa->data[i] = 'A' + i;  // 填充数据
        }
    
        // 打印柔性数组
        printf("柔性数组内容:");
        for (int i = 0; i < fa->length; i++) {
            printf("%c ", fa->data[i]);
        }
        printf("\n");
    
        // 释放内存
        free(fa);
    
        return 0;
    }
    

内存分布分析:

假设 sizeof(struct FlexibleArray) 为 4 字节(假设 int 占 4 字节):

  1. malloc 分配的内存大小为:4 + n * sizeof(char),即 4 + 10 = 14 字节。
  2. length 占用固定的 4 字节。
  3. data[] 根据需要动态分配 10 字节,存储数组数据。

 


4. 为什么使用柔性数组?

优势:

  1. 节省内存:避免声明固定大小的数组。例如,如果某个结构体只需要存储一个动态长度的数组,柔性数组能够根据实际需要分配内存,而不是浪费空间。
  2. 灵活性强:适合实现复杂的数据结构,如变长消息或动态数组。

与固定大小数组的比较:

使用固定大小的数组会导致内存浪费,例如:

struct FixedArray {
    int length;
    char data[100];  // 固定大小的数组
};

即使只需要存储 10 个字符,数组仍然占用 100 字节。而柔性数组则根据需要分配内存,节省了 90 字节的空间。

 

5. 常见的使用场景

场景1:动态消息结构

在网络通信或协议解析中,消息的长度通常是可变的。可以使用柔性数组实现动态消息结构:

struct Message {
    int id;         // 消息ID
    int length;     // 消息内容长度
    char content[]; // 消息内容
};

分配内存和使用:

struct Message *msg = malloc(sizeof(struct Message) + msg_length);
msg->id = 1;
msg->length = msg_length;
memcpy(msg->content, data, msg_length);

场景2:实现动态数组

柔性数组可以作为动态数组的基础,实现灵活的动态大小。


6. 使用柔性数组的注意事项

  1. 内存分配: 使用 malloccalloc 为结构体分配内存时,必须计算柔性数组的实际大小。
  2. 不能直接使用 sizeof 柔性数组的大小不会包含在 sizeof 结果中。例如:
struct FlexibleArray {
    int length;
    char data[];
};
printf("%lu\n", sizeof(struct FlexibleArray)); // 只计算固定部分,不包含柔性数组

 

建议优先使用 柔性数组,以保证代码的兼容性和可维护性。

  1. 释放内存: 柔性数组的内存是动态分配的,释放时必须确保调用 free
  2. 柔性数组只能在结构体中使用: 不能单独定义柔性数组,例如:
    char arr[];  // 非法,柔性数组只能作为结构体成员
    

    7. 柔性数组与 GCC 零长度数组的区别

    在某些编译器(如 GCC)中,零长度数组 char data[0]; 被用于类似的用途。但标准 C99 中推荐使用柔性数组 char data[];

  3. 柔性数组(C99 标准):
    • 语法:char data[];
    • 优势:符合标准,更安全。
  4. 零长度数组(GCC 扩展):
    • 语法:char data[0];
    • 不符合 C 标准,但在某些旧代码中广泛使用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/920404.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

图像处理实验报告

实验一 图像处理的MATLAB基础 实验目的&#xff1a;熟悉数字图象处理的基本软件工具和操作 实验内容&#xff1a;Matlab应用复习&#xff0c;矩阵产生、操作&#xff1b;矩阵运算以及字符运算。 1.利用增量产生向量[0,2,4,6,8,10]。 2.利用magic(n)函数产生7维魔鬼矩阵A&am…

NLP论文速读(MPO)|通过混合偏好优化提高多模态大型语言模型的推理能力

论文速读|Dynamic Rewarding with Prompt Optimization Enables Tuning-free Self-Alignment of Language Models 论文信息&#xff1a; 简介&#xff1a; 本文探讨的背景是多模态大型语言模型&#xff08;MLLMs&#xff09;在多模态推理能力上的局限性&#xff0c;尤其是在链式…

如何更改手机GPS定位

你是否曾想过更改手机GPS位置以保护隐私、玩游戏或访问受地理限制的内容&#xff1f;接下来我将向你展示如何使用 MagFone Location Changer 更改手机GPS 位置&#xff01;无论是在玩Pokmon GO游戏、发布社媒贴子&#xff0c;这种方法都快速、简单且有效。 第一步&#xff1a;下…

Java项目实战II基于微信小程序的图书馆自习室座位预约平台(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、文档参考 五、核心代码 六、源码获取 全栈码农以及毕业设计实战开发&#xff0c;CSDN平台Java领域新星创作者&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。 一、前言 在知识爆炸的时代&#xff0c;图书馆和…

图像处理 - 色彩空间转换

色彩空间转换的含义与原理 色彩空间转换是指将一种颜色模型或表示方式中的颜色数据映射到另一种颜色模型中的过程。色彩空间&#xff08;Color Space&#xff09;本质上是一个三维坐标系统&#xff0c;每个点都表示图像中的某种颜色。在实际应用中&#xff0c;由于不同的色彩空…

web——sqliabs靶场——第十三关——报错注入+布尔盲注

发现是单引号加括号闭合的 尝试联合注入 发现不太行&#xff0c;那尝试报错注入。 测试报错注入 unameadmin) and updatexml(1,0x7e,3) -- &passwdadmin&submitSubmit 爆数据库 unameadmin) and updatexml(1,concat(0x7e,database(),0x7e),3) -- &passwdadmin&a…

大语言模型中Softmax函数的计算过程及其参数描述

文章目录 概要Softmax的定义 概要 **Softmax 作用&#xff1a;**主要用于自注意层&#xff0c;作用是将输出映射成区间在(0,1)的值&#xff0c;并且做了归一化&#xff0c;所有元素的和累加起来等于1。 **Softmax使用场景&#xff1a;**Llama-7B模型中&#xff0c;其在自注意层…

Jmeter中的断言(三)

9--MD5Hex断言 功能特点 数据完整性验证&#xff1a;验证响应数据的 MD5 哈希值是否符合预期。简单配置&#xff1a;只需提供预期的 MD5 哈希值即可。灵活配置&#xff1a;可以设置多个断言条件&#xff0c;满足复杂的测试需求。 配置步骤 添加 MD5Hex 断言 右键点击需要添加…

后端开发如何高效使用 Apifox?

对于后端开发者来说&#xff0c;日常工作中少不了接口的设计、调试和文档编写。你是否也曾因接口文档更新不及时、测试工具分散而头疼不已&#xff1f;Apifox&#xff0c;这款全能型工具&#xff0c;或许能成为你的效率神器&#xff01; Apifox究竟有哪些功能能帮助后端开发者…

JMeter监听器与压测监控之 InfluxDB

1. 简介 在本文中&#xff0c;我们将介绍如何在 Kali Linux 上通过 Docker 安装 InfluxDB&#xff0c;并使用 JMeter 对其进行性能监控。InfluxDB 是一个高性能的时序数据库&#xff0c;而 JMeter 是一个开源的性能测试工具&#xff0c;可以用于对各种服务进行负载测试和性能监…

操作系统基础——针对实习面试

目录 操作系统基础什么是操作系统&#xff1f;简述其主要功能请举例几种不同类型的操作系统&#xff0c;并简要说明它们的特点 操作系统基础 什么是操作系统&#xff1f;简述其主要功能 一、操作系统的定义 操作系统&#xff08;Operating System&#xff0c;简称OS&#xff…

uni-app快速入门(十)--常用内置组件(下)

本文介绍uni-app的textarea多行文本框组件、web-view组件、image图片组件、switch开关组件、audio音频组件、video视频组件。 一、textarea多行文本框组件 textarea组件在HTML 中相信大家非常熟悉&#xff0c;组件的官方介绍见&#xff1a; textarea | uni-app官网uni-app,un…

CSS中Flex布局应用实践总结

① 两端对齐 比如 要求ul下的li每行四个&#xff0c;中间间隔但是需要两段对齐&#xff0c;如下图所示&#xff1a; 这是除了基本的flex布局外&#xff0c;还需要用到:nth-of-type伪类来控制每行第一个与第四个的padding。 .hl_list{width: 100%;display: flex;align-items…

Spring Boot + Vue 基于 RSA 的用户身份认证加密机制实现

Spring Boot Vue 基于 RSA 的用户身份认证加密机制实现 什么是RSA&#xff1f;安全需求介绍前后端交互流程前端使用 RSA 加密密码安装 jsencrypt库实现敏感信息加密 服务器端生成RSA的公私钥文件Windows环境 生成rsa的公私钥文件Linux环境 生成rsa的公私钥文件 后端代码实现返…

Three.js + AI:AI 算法生成 3D 萤火虫飞舞效果~

AI 驱动 3D 动画 大家好&#xff0c;我是石小石&#xff01;随着 Web 技术的发展&#xff0c;Three.js 成为构建 3D 图形和动画的主流工具。与此同时&#xff0c;人工智能&#xff08;AI&#xff09;在图像处理、动作生成等领域表现出强大能力。将 AI 与 Three.js 结合&#x…

bpmn.js显示流程图

目标&#xff1a;vue2 接口返回xml - 弹窗显示流程图 - 根据需求高亮节点 一、安装依赖 npm i bpmn-js 添加上的版本是 "bpmn-js": "^11.5.0" 二、只读显示流程图 <div ref"canvas" style"width:100%;height: calc(100vh…

Vue Form表单的使用,rules格式校验网络校验,键盘按键监听

Form表单 rules格式校验 可以在validator中进行网络请求&#xff0c;实现网络校验 const formRules {userName: [{required: true, message: "用户名不能为空", trigger: blur}, {min: 5,max: 10,message: "长度必须5-10位",trigger: blur}],passWord: …

单片机智能家居火灾环境安全检测-分享

目录 前言 一、本设计主要实现哪些很“开门”功能&#xff1f; 二、电路设计原理图 电路图采用Altium Designer进行设计&#xff1a; 三、实物设计图 四、程序源代码设计 五、获取资料内容 前言 传统的火灾报警系统大多依赖于简单的烟雾探测器或温度传感器&#xff0c;…

TCP并发服务器

端口号快速复用函数 通过getsockopt和setsockopt函数&#xff0c;管理套接字的端口号复用设置。具体操作如下&#xff1a; getsockopt函数 int getsockopt(int sockfd, int level, int optname, void *optval, socklen_t *optlen);功能&#xff1a;获取套接字的某些选项的属性。…

vue3的宏到底是什么东西?

前言 从vue3开始vue引入了宏&#xff0c;比如defineProps、defineEmits等。我们每天写vue代码时都会使用到这些宏&#xff0c;但是你有没有思考过vue中的宏到底是什么&#xff1f;为什么这些宏不需要手动从vue中import&#xff1f;为什么只能在setup顶层中使用这些宏&#xff…