跟我学C++中级篇——零长度数组

2024-05-02 22:12

本文主要是介绍跟我学C++中级篇——零长度数组,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、零长度数组

最初使用零长度数组是在串口通信和网络通信中使用的,它类似于下面的代码:

struct Data
{int a;char c;char buf[0];
};

零长度数组又叫柔性数组,这个非常贴切。它早期是GNU的一个扩展,后来在C99中也进行了支持(-pedantic 禁用扩展),不过在C99中使用时,就不用写那个0了。如果在早期的VC或者其它平台上不支持这种零长度数组的,可以考虑使用一些变通的方式。

二、应用场景

零长度数组主要用来对一些不确定长度的通信负载的内容进行接收。在C/C++的编码过程中,有一个小的技巧,即可以把char*指针强转到结构体的指针,这样就可以直接操作结构体。这对于网络或者串口等通信中有着很强的优势(需要处理一下字节对齐)。

那么有人会问为什么不使用指针呢?或者给一个很长的固定数组。首先,给定一个很长的数组,到底多长是很长?如果能确定最多不过一百个,那么用一百个长度来取代零长度数组可能会更好,但可能跨度很大呢,从几个字节长度到几兆字节长度呢?有人可能说,谁会设计这种协议呢?可有的时候儿协议不是自己设计的,用别人的怎么办?那么这时候儿用零长度指针就非常好用。
那为什么不用指针呢?学习过编译相关知识的都知道,指针指向的地址空间,一般情况下是不会和指针当前的地址连续的,以上面的Data结构体为例,如果创建一个对象,使用零长度数组,所有的字节其实是从变量a依次按顺序向下排列的,是线性的。但如果把buf定义成一个指针,那么buf本身这个地址是排在最后的,但对应的指针指向的空间,则指向了另外一个空间地址。表面上创建了一个Data对象,但其是有两块地址空间被分配,这两块地址空间并不连续。不连续意味着什么?意味着上面提到的将char*强转成指定的结构体指针成为了不可能。因为buf得到的地址空间,未必是原来定义的空间,这个在硬件通信中非常明显。
最好的方法就是,直接传输一大段数据,这些数据是连续的存储在一起的,那么通过强转就可直接进行数据操作,可数据长度又是未知的,而普通的数组长度必须是提前指定的,这就是一个矛盾,所以零长度数组的优势就体现出来了。它其实相当于一个占位符,告诉空间分配者,这后面还有一大堆的数据,至于多长,根据实际情况来定。零长度数组等于是取了个巧,既符合了标准中的数组长度必须提前固定,又实现了指针的变长分配。

三、例程

下面看一个模拟的例程:

struct DataWater {int count;float size;char buf[0];
};
char dbuf[1024] = {0};
void testZeroArray(char *data) {DataWater *dw;dw = (DataWater *)data;std::cout << dw->count << std::endl;
}
int main() {testZeroArray(dbuf);return 0;
}

假如testZeroArray函数是一个接收串口通信或者网络通信的协议结构体,而data就是通过接收得来的一个字节流,那么这个函数的代码就可以直接解析出来协议的内容并为程序所用。

四、总结

其实零长度数组是一个技巧,这样的小技巧在C/C++中有很多。它们在一些特定的场景下有着非常好的应用,不过前面的定语一定要记清楚,特定的场景。这也是C/C++让初学者感到难缠的一个重要原因。到处都有一些特殊情况,而特殊情况里可能又套着特殊情况。
这种小技巧的东西,不用刻意学习,用到了,想起来有,查查资料用就可以了。

这篇关于跟我学C++中级篇——零长度数组的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/955246

相关文章

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que