我在使用供应商 API 从设备检索数据时遇到问题。供应商文档中建议的调用的工作版本如下所示:
int dataArray[15][20];
getData(15, 20, (int*) dataArray);
调用签名描述为:
void getData(xSize, ySize, int*);
我有兴趣使用此函数将数据复制到具有动态大小的连续整数数组中,如下所示:
int *flatDataArray = (int*) malloc(xSize * ySize * sizeof(int));
我很难正确调用,而且段错误有点难以调试 :(
我认为我的误解在于 int 数组的表示方式,以及将其转换为 (int*) 时会发生什么。在这个例子中是 int** 类型的“dataArray”,还是其他什么?如果是这样,您是否必须构造一个指向 flatDataArray 的指针数组才能使上述函数正常工作?
最佳答案
由于转换,第一行“有效”。
给定:
int dataArray[15][20];
getData(15, 20, (int*) dataArray);
dataArray
单机的表达式值类型为int (*)[20]
。指向 20 个 int
数组的指针。这是有道理的,因为根据 C 标准,数组的表达式值是其第一个元素的地址和指向相同类型的指针。嗯,数组数组的第一个“元素”(dataArray
是什么)是 int[20]
类型,指向类型的指针给了我们int(*)[20]
。
也就是说,相同的规则适用于该数组数组中的第一个数组。简而言之,这将起作用:
int dataArray[15][20];
getData(15, 20, dataArray[0]);
就像 dataArray
是一个表达式,结果是 int (*)[20]
类型的第一个元素(数组)的地址,其中的第一个数组数组同样有一个表达式值,它是它的第一个元素的地址,一个int
,与该地址关联的类型是int *
As碰巧,这是数组数组的第一个数组的第一个元素。在其运行的底层系统的线性内存背景中,它最终都解析为相同的地址(dataArray
位于内存中)。它是与该地址相关联的类型,这取决于它是如何设计的,是不同的。然而,无论类型如何,底层内存背景都是相同的:一个连续的 300 int
序列。以下所有返回的地址将相同;只有类型不同(并在评论中注明)
&dataArray // int (*)[15][20]
dataArray // int (*)[20]
dataArray[0] // int *
&dataArray[0][0] // int *
不,这些不是唯一的组合。我至少留下了一个。看看您是否能找出缺少的内容。
无论如何,按照你在这里的方式进行分配:
int *flatDataArray = malloc(xSize * ySize * sizeof(int));
之所以有效,是因为您只是简单地构建了一个 xSize by ySize
线性背景。因为这是 C,下面的也可以使用,利用语言的 VLA(可变长度数组)特性:
int (*arr2D)[ySize] = malloc(xSize * sizeof(*arr2D));
假设 xSize
是行的数量,ySize
是您寻找的列的数量(我似乎有一半时间都弄错了,这就是为什么我更喜欢绰号“行”和“列”)。上面的分配说“给我分配 xSize
个 int[ySize]
的空间。”。在代码中的好处是你可以像处理二维数组一样处理这个问题(这正是它的本质):
arr2d[i][j] = value;
对于 0..(xsize-1)
中的任何 i
和 0..(ysize-1) 中的
,就像你会/可以像你正常声明它一样。这个(VLAs)是 C 做得很好而 C++ 做不到的一件事(但是话又说回来,C++ 中有很多容器可以首先以不同的方式解决这个问题,所以这真的不公平比较)。j
祝你好运。
关于cArray[][] 指针数组转换,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24372246/