我想创建一个C函数,它将二维2D数组作为参数,并通过索引对该数组进行操作,例如: printf("%f ", array[i][j])
。
我能够从各种例子和SO问题中拼凑出来的东西是这样的:
void printArray(double **array, int m, int n)
{
int i, j;
for (i = 0; i < m; i++)
{
for (j = 0; j < n; j++)
{
printf("%f ", array[i][j]);
}
printf("\n");
}
}
在main
中,我能够成功打印数组:
int i, j, k = 0, m = 5, n = 6;
double **a = malloc(m * sizeof(*a));
//Initialize the arrays
for (i = 0; i < m; i++)
{
a[i] = malloc(n * sizeof(*(a[i])));
}
for (i = 0; i < m; i++)
{
for (j = 0; j<n; j++)
{
k++;
a[i][j] = k;
}
}
printArray(a, m, n);
但是,当我尝试将数组初始化为某个给定值然后调用:
double a[5][6] = { { 1, 2, 3, 4, 5 ,6},
{ 1, 2, 3, 4, 5, 6},
{ 1, 2, 3, 4, 5, 6},
{ 1, 2, 3, 4, 5, 6},
{ 1, 2, 3, 4, 5, 6} };
printArray(a, 5, 6);
我遇到以下错误:
Unhandled exception at 0x011514D3 in Example.exe: 0xC0000005:
Access violation reading location 0xA1F8E3AC.
有人可以解释我的错误是什么以及如何解决它? 编辑
请注意,出于函数定义的目的,我将在运行时知道数组的大小,但不知道编译时的大小。此外,我正在使用VisualStudio 2013在Windows上进行编译。
答案 0 :(得分:5)
double a[5][6]
的类型为double[][6]
,与double**
不同。 double**
是指向double的指针。 double[][6]
是编译器管理的数据类型,表示二维数组。
此处发生的事情是,您创建的double[][6]
在double**
的调用中被静默投放到printArray
。
如果您的功能需要double**
,则需要传递double**
。您可以通过单独填充每个成员数组来初始化数组内容:
double row1[3] = {3, 4, 5}; a[1] = row1;
这解决了这个问题;因为编译器将double[]
存储为double
值的连续数组,所以隐式将其double*
投射到上面是安全的。
另一种解决方案是改变功能以采取真实的&#34; double[][6]
,而不是指针指针。你如何使用非固定大小的数组来达到你特定的C品牌;据我所知,它不属于C标准的一部分。
第三种解决方案是使用malloc
逐行构建数组,并使用a[0][0] = 1
逐个单元格填充它,依此类推。您已经在问题中使用了这个,并且它可以正常工作。
最后要注意的是你在堆栈上分配a
:当你的main
函数退出时,访问它将导致未定义的行为。