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

柔性数组(变长数组)介绍

柔性数组简介

柔性数组,或称为可变长度数组,是一种在C语言结构体定义中使用的特殊数组,它允许结构体拥有一个可变大小的数组成员。柔性数组成员必须是结构体的最后一个成员,且它不占用结构体大小的计算,这使得可以动态地分配超出结构体声明大小的内存,从而容纳变长的数据。

优点

  • 动态内存管理:使用柔性数组可以根据需要动态地分配更多的内存,这在处理不确定大小的数据时非常有用。
  • 内存连续性:柔性数组的数据存储在单一连续的内存块中,这有利于提高内存访问效率。
  • 简化指针操作:通过减少额外的指针或分配,柔性数组可以简化代码和降低出错率。

如何使用柔性数组

要在C语言中使用柔性数组,你需要在结构体定义中将最后一个元素声明为未指定大小的数组。这里是一个典型的使用示例:

#include <stdio.h>
#include <stdlib.h>typedef struct {int length;double data[]; // 柔性数组成员
} FlexibleArray;int main() {int desiredLength = 5;// 分配内存时,包括结构体基础大小和数组所需的额外空间FlexibleArray *array = (FlexibleArray *)malloc(sizeof(FlexibleArray) + sizeof(double) * desiredLength);array->length = desiredLength;for (int i = 0; i < array->length; i++) {array->data[i] = i * 2.0;}for (int i = 0; i < array->length; i++) {printf("Element %d = %f\n", i, array->data[i]);}free(array);return 0;
}

注意事项

  • 柔性数组成员不占用结构体大小的计算。
  • 只有位于结构体最后一个成员位置的数组可以被声明为柔性数组。
  • 分配含有柔性数组的结构体时,需要手动计算额外内存的需求。
  • 使用完毕后,需要手动释放内存以避免内存泄露。

在网络通信中使用柔性数组

消息结构

// message.h
#ifndef MESSAGE_H
#define MESSAGE_H#include <stdint.h>typedef struct {uint32_t length; // 消息长度char data[0];    // 数据部分
} message_t;#endif

服务端

// server.c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include "message.h"int main(int argc, char *argv[]) {int sockfd, newsockfd, portno;socklen_t clilen;char buffer[256];struct sockaddr_in serv_addr, cli_addr;int n;if (argc < 2) {fprintf(stderr, "ERROR, no port provided\n");exit(1);}sockfd = socket(AF_INET, SOCK_STREAM, 0);if (sockfd < 0) printf("ERROR opening socket\n");bzero((char *) &serv_addr, sizeof(serv_addr));portno = atoi(argv[1]);serv_addr.sin_family = AF_INET;serv_addr.sin_addr.s_addr = INADDR_ANY;serv_addr.sin_port = htons(portno);if (bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0) printf("ERROR on binding\n");listen(sockfd, 5);clilen = sizeof(cli_addr);newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);if (newsockfd < 0) printf("ERROR on accept\n");uint32_t msg_length;n = read(newsockfd, &msg_length, sizeof(msg_length));if (n < 0) printf("ERROR reading from socket\n");msg_length = ntohl(msg_length); // 确保网络字节序转换为主机字节序message_t *msg = (message_t*)malloc(sizeof(message_t) + msg_length - 1); // 分配额外的空间n = read(newsockfd, msg->data, msg_length);if (n < 0) printf("ERROR reading from socket\n");printf("Here is the message: %s\n", msg->data);close(newsockfd);close(sockfd);return 0; 
}

客户端

// client.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h> 
#include "message.h"
int main(int argc, char *argv[]) {int sockfd, portno, n;struct sockaddr_in serv_addr;struct hostent *server;char buffer[256];if (argc < 3) {fprintf(stderr,"usage %s hostname port\n", argv[0]);exit(0);}portno = atoi(argv[2]);sockfd = socket(AF_INET, SOCK_STREAM, 0);if (sockfd < 0) printf("ERROR opening socket\n");server = gethostbyname(argv[1]);if (server == NULL) {fprintf(stderr,"ERROR, no such host\n");exit(0);}bzero((char *) &serv_addr, sizeof(serv_addr));serv_addr.sin_family = AF_INET;bcopy((char *)server->h_addr, (char *)&serv_addr.sin_addr.s_addr,server->h_length);serv_addr.sin_port = htons(portno);if (connect(sockfd,(struct sockaddr *) &serv_addr,sizeof(serv_addr)) < 0) printf("ERROR connecting\n");printf("Please enter the message: ");bzero(buffer,256);fgets(buffer,255,stdin);uint32_t msg_length = strlen(buffer);// 发送消息长度uint32_t n_msg_length = htonl(msg_length); // 转换为网络字节序write(sockfd, &n_msg_length, sizeof(n_msg_length));// 发送消息数据n = write(sockfd, buffer, msg_length);if (n < 0) printf("ERROR writing to socket\n");close(sockfd);return 0;
}

相关文章:

柔性数组(变长数组)介绍

柔性数组简介 柔性数组&#xff0c;或称为可变长度数组&#xff0c;是一种在C语言结构体定义中使用的特殊数组&#xff0c;它允许结构体拥有一个可变大小的数组成员。柔性数组成员必须是结构体的最后一个成员&#xff0c;且它不占用结构体大小的计算&#xff0c;这使得可以动态…...

AMS、PMS和WMS学习链接

原文: Framework学习&#xff08;三&#xff09;之PMS、AMS、WMS_ams pms-CSDN博客 1:PackageMangerService&#xff08;PMS&#xff09;讲解博主 PMS系列我觉得csdn博主jeanboy讲的非常好&#xff0c;这里附上博主的博客链接jeanboy。这是一位资深级的博客专家。关于他PMS的讲…...

typedef 在枚举类型enum的使用方式

enum类型用途:为程序中的一组相关的常量取名字,以便以程序的可读性和维护性 方式一:typedef enum 变量名{E_XXXXX = 0,E_xxxx,E_XXXX_MAX }变量名_e; 方式二: typedef enum {E_xxxx = 0,E_xxxx,E_xxx_MAX}变量名_e;#include <stdio.h> typedef enum vii_vpp_mode {E_…...

DDD领域模型驱动

传统MVC架构 DDD架构: api层:api请求方式,透传【传递参数】,几个业务对应api 业务层:做编排,业务里要有哪些服务,执行顺序是什么,以及怎么做 领域层:负责领域内调用,然后领域怎么划分 Dao层:数据库操作【或者另外一个应用 数据源之类的】 遵守原则: ①允许跨层…...

基于pytest的证券清算系统功能测试工具开发

需求 1.造测试数据&#xff1a;根据测试需要&#xff0c;自动化构造各业务场景的中登清算数据与清算所需起来数据 2.测试清算系统功能&#xff1a; 自动化测试方案 工具设计 工具框架图 工具流程图 实现技术 python, pytest, allure, 多进程&#xff0c;mysql, 前端 效果 测…...

土地利用数据分类过程教学/土地利用分类/遥感解译/土地利用获取来源介绍/地理数据获取

本篇主要介绍如何对影像数据进行分类解译&#xff0c;及过程教学&#xff0c;示例数据下载链接&#xff1a;数据下载链接 一、背景介绍 土地是人类赖以生存与发展的重要资源和物质保障&#xff0c;在“人口&#xff0d;资源&#xff0d;环境&#xff0d;发展&#x…...

图【数据结构】

文章目录 图的基本概念邻接矩阵邻接表图的遍历BFSDFS 图的基本概念 图是由顶点集合及顶点间的关系组成的一种数据结构 顶点和边&#xff1a;图中结点称为顶点 权值:边附带的数据信息 路径 &#xff1a; 简单路径 和 回路&#xff1a; 子图&#xff1a;设图G {V, E}和图G1…...

整块代码自动生成、智能括号匹配……CodeGeeX编程提效,功能再升级!

CodeGeeX插件功能持续打磨&#xff0c;希望成为开发者更高效的智能编程工具&#xff0c;提高开发速度和代码质量。今天介绍VSCode中最新的v2.4.0版本插件新功能&#xff0c;让你在编写代码时更加得心应手。 一、新增block代码块生成的设置 CodeGeeX插件中&#xff0c;以往针对…...

java实现计算ROUGE-L指标(一)

ROUGE (Recall-Oriented Understudy for Gisting Evaluation) 是用于评估自动文摘或机器翻译的一种评估方法&#xff0c;其中的ROUGE-L指标是基于最长公共子序列&#xff08;Longest Common Subsequence&#xff0c;LCS&#xff09;来计算的 我们做AI问答系统&#xff0c;需要一…...

LLM之RAG实战(二十九)| 探索RAG PDF解析

对于RAG来说&#xff0c;从文档中提取信息是一种不可避免的场景&#xff0c;确保从源文件中提取出有效的内容对于提高最终输出的质量至关重要。 文件解析过程在RAG中的位置如图1所示&#xff1a; 在实际工作中&#xff0c;非结构化数据比结构化数据丰富得多。如果这些海量数据无…...

C while 和 do while 区别

while 和 do while 都是 C 语言中的循环语句&#xff0c;它们的主要区别在于循环体执行的顺序。 while 循环首先检查循环条件&#xff0c;只有当条件为真时才执行循环体。因此&#xff0c;如果条件一开始就为假&#xff0c;那么循环体将永远不会执行。而如果条件一直为真&…...

力扣每日一题 在受污染的二叉树中查找元素 哈希 DFS 二进制

Problem: 1261. 在受污染的二叉树中查找元素 思路 &#x1f468;‍&#x1f3eb; 灵神题解 &#x1f496; 二进制 时间复杂度&#xff1a;初始化为 O ( 1 ) O(1) O(1)&#xff1b;find 为 O ( m i n ( h , l o g 2 t a r g e t ) O(min(h,log_2target) O(min(h,log2​targ…...

安卓Java面试题 91- 100

91. 请描述一下Intent 和 IntentFilter ?Intent是组件的通讯使者,可以在组件间传递消息和数据。 IntentFilter是intent的筛选器,可以对intent的action,data,catgory,uri这些属性进行筛选,确定符合的目标组件🚀🚀🚀🚀🚀🚀92. 阐述什么是IntentService?有何优…...

BM1684X搭建sophon c++环境

1:首先安装编译好sophon-sail 比特大陆BM1684X开发环境搭建--SOC mode-CSDN博客 2:在将之前配置的soc-sdk拷贝一份到sdk根目录&#xff0c;将交叉编译好的sail中的build_soc拷贝至soc-sdk文件夹内&#xff1b; cp -rf build_soc/sophon-sail/inlcude soc-sdk cp -rf build_soc…...

UDP通讯测试

参考资料:UNIX网络编程 实验平台:PC为client,RaspberryPi为server 基本类型和接口函数,参考man手册 #include <sys/socket.h>struct sockaddr {sa_family_t sa_family; /* Address family */char sa_data[]; /* Socket address */};#inclu…...

Linux - 进程间通信

1、进程间通信介绍 1.1、进程间通信目的 数据传输&#xff1a;一个进程需要将它的数据发送给另一个进程&#xff1b;资源共享&#xff1a;多个进程之间共享同样的资源&#xff1b;通知事件&#xff1a;一个进程需要向另一个或一组进程发送消息&#xff0c;通知它&#xff08;…...

代码随想录算法训练营第七天|454. 四数相加 II

454. 四数相加 II 已解答 中等 相关标签 相关企业 给你四个整数数组 nums1、nums2、nums3 和 nums4 &#xff0c;数组长度都是 n &#xff0c;请你计算有多少个元组 (i, j, k, l) 能满足&#xff1a; 0 < i, j, k, l < nnums1[i] nums2[j] nums3[k] nums4[l] 0 示例 …...

蓝桥杯刷题(五)

[蓝桥杯 2022 省 B] 刷题统计 题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 提示题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 样例 #2样例输入 #2样例输出 #2 提示 题目描述 小明决定从下周一开始努力刷题准备蓝桥杯竞赛。他计划周一至周五每天做 a a…...

mysql语句中想要查询某一月每一天日期的平均值 ,SSM框架如何实现

mysql语句中想要查询某一月每一天日期的平均值 为了查询某一月份每一天的平均值&#xff0c;你可以使用以下SQL查询语句。这里假设你有一个表格data_table&#xff0c;它有一个日期时间列date_column和一个需要计算平均值的数值列value_column。 SELECTDATE_FORMAT(date_colum…...

前端框架的发展历程

文章目录 前言 一、静态页面时代 二、JavaScript的兴起 三、jQuery的出现 四、前端框架的崛起 1.AngularJS 2.React 3.Vue.js 五、面向组件化的发展趋势 总结 前言 前端框架的发展史就是一个不断进化的过程&#xff0c;它的发展和进化一定程度…...

单电阻采样 基于单电阻采样的相电流重构算法 keil完整工程。 单电阻采样 f103的单电阻...

单电阻采样 基于单电阻采样的相电流重构算法 keil完整工程。 单电阻采样 f103的单电阻&#xff0c;完整工程&#xff0c;带文档&#xff0c;带硬件资料。 f3平台的单电阻完整工程&#xff0c;代码详细注释。 还有微芯的单电阻smo代码加文档 具体如截图请看下一、工程概述 本工程…...

百度网盘秒传链接全平台解决方案:告别漫长等待,实现文件瞬间转移

百度网盘秒传链接全平台解决方案&#xff1a;告别漫长等待&#xff0c;实现文件瞬间转移 【免费下载链接】baidupan-rapidupload 百度网盘秒传链接转存/生成/转换 网页工具 (全平台可用) 项目地址: https://gitcode.com/gh_mirrors/bai/baidupan-rapidupload 你是否曾因…...

Adrenaline终极指南:解锁PSP模拟器的完整潜力

Adrenaline终极指南&#xff1a;解锁PSP模拟器的完整潜力 【免费下载链接】Adrenaline Custom Firmware 6.61 Adrenaline for the PSP Emulator 项目地址: https://gitcode.com/gh_mirrors/adr/Adrenaline 你是否曾为PSP模拟器的功能限制而烦恼&#xff1f;想要在PS Vit…...

四足机器人步态调参实战:如何用Walk These Ways控制器实现楼梯穿越与抗干扰行走

四足机器人步态调参实战&#xff1a;Walk These Ways控制器在复杂地形中的应用技巧 当Unitree Go1机器人第一次站在楼梯前时&#xff0c;开发者们面临着一个经典困境——如何让这台在平地上表现优异的机器跨越这道障碍。传统解决方案往往需要重新训练模型或调整底层算法&#…...

DepotDownloader核心功能解析:从App下载到工作坊内容获取的完整指南

DepotDownloader核心功能解析&#xff1a;从App下载到工作坊内容获取的完整指南 【免费下载链接】DepotDownloader Steam depot downloader utilizing the SteamKit2 library. 项目地址: https://gitcode.com/gh_mirrors/de/DepotDownloader DepotDownloader是一款功能强…...

迪卡侬集团2025年净销售额同比增长4.0%至168亿欧元

、美通社消息&#xff1a;2025年&#xff0c;面对瞬息万变的市场环境&#xff0c;迪卡侬集团展现出稳健的经营韧性与持续的战略定力&#xff0c;整体经营表现稳步向好&#xff0c;以当地货币计算&#xff0c;商品交易总额(GMV)同比增长7.1%&#xff0c;净销售额同比增长5.6%&am…...

StructBERT语义相似度工具快速体验:输入句子秒出结果

StructBERT语义相似度工具快速体验&#xff1a;输入句子秒出结果 1. 工具简介与核心价值 当你需要快速判断两段中文文字是否表达相同含义时&#xff0c;传统方法往往需要人工逐字比对或依赖复杂的算法配置。现在&#xff0c;基于StructBERT-Large模型的语义相似度工具让这个过…...

Spring Boot后端实战:手把手教你处理Google Play订阅续费、降级与退款回调

Spring Boot实战&#xff1a;Google Play订阅状态变更的深度处理指南 订阅业务中的关键挑战 移动应用订阅模式已成为开发者重要的收入来源&#xff0c;而Google Play作为全球最大的应用分发平台&#xff0c;其订阅系统的复杂性往往让开发者头疼。特别是当用户进行订阅续费、降…...

Ansible Roles深度指南:如何像搭积木一样管理复杂Playbook?

Ansible Roles架构设计&#xff1a;构建企业级配置管理的乐高积木 在电商系统多环境部署的复杂场景中&#xff0c;开发团队经常面临这样的困境&#xff1a;测试环境的配置意外污染了生产环境&#xff0c;不同服务间的变量命名冲突导致部署失败&#xff0c;或者新增服务器时需要…...

新站如何运用SEO手段快速上首页_网站链接建设有助于SEO快速排名吗

新站如何运用SEO手段快速上首页 在互联网时代&#xff0c;新站如何快速上首页成为了许多网站创始人和SEO专业人士的共同关注点。快速攀升到搜索引擎的首页不仅能带来大量流量&#xff0c;还能提升品牌知名度。新站应如何运用SEO手段来实现这一目标呢&#xff1f;本文将从多个角…...