2

如何在 C 或 C++ 中通过引用传递未知大小的多维数组?

编辑:

例如,在 main 函数中,我有:

int main(){
    int x, y;
    int arr[x][y];
    // pass_by_ref(/* passing just arr[][] by reference */);
}  

和功能:

void pass_by_ref(/* proper parameter for arr[][] */){
    // int size_x_Arr = ???
    // int size_y_arr = ???
}

如何实现注释行?

4

5 回答 5

4

简单地说,你不能。在 C 中,您不能通过引用传递,因为 C 没有引用。在 C++ 中,您不能传递大小未知的数组,因为 C++ 不支持可变长度数组。

替代解决方案:在 C99 中,将指针传递给可变长度数组;在 C++ 中,传递对std::vector<std::vector<T>>.

C99 演示:

#include <stdio.h>

void foo(int n, int k, int (*arr)[n][k])
{
    int i, j;
    for (i = 0; i < n; i++) {
        for (j = 0; j < k; j++) {
            printf("%3d ", (*arr)[i][j]);
        }
        printf("\n");
    }
}

int main(int argc, char *argv[])
{
    int a = strtol(argv[1], NULL, 10);
    int b = strtol(argv[2], NULL, 10);

    int arr[a][b];
    int i, j;
    for (i = 0; i < a; i++) {
        for (j = 0; j < b; j++) {
            arr[i][j] = i * j;
        }
    }

    foo(a, b, &arr);

    return 0;
}

C++03 演示:

#include <iostream>
#include <vector>
#include <cstdlib>
#include <ctime>

void foo(std::vector < std::vector < int > > &vec)
{
    for (std::vector < std::vector < int > >::iterator i = vec.begin(); i != vec.end(); i++) {
        for (std::vector<int>::iterator j = i->begin(); j != i->end(); j++) {
            std::cout << *j << " ";
        }
        std::cout << std::endl;
    }
}

int main(int argc, char *argv[])
{
    int i = strtol(argv[1], NULL, 10);
    int j = strtol(argv[2], NULL, 10);

    srand(time(NULL));

    std::vector < std::vector < int > > vec;
    vec.resize(i);
    for (std::vector < std::vector < int > >::iterator it = vec.begin(); it != vec.end(); it++) {
        it->resize(j);
        for (std::vector<int>::iterator jt = it->begin(); jt != it->end(); jt++) {
            *jt = random() % 10;
        }
    }

    foo(vec);

    return 0;
}
于 2013-03-06T11:00:15.410 回答
3

H2CO3 的解决方案适用于支持 VLA 的 C99 或 C2011 编译器。对于支持 VLA 的 C89 或 C2011 编译器,或者(上帝保佑)K&R C 编译器,您必须执行其他操作。

假设您要传递一个连续分配的数组,您可以传递一个指向第一个元素( &a[0][0]) 的指针以及维度大小,然后将其视为一维数组,映射索引如下:

void foo( int *a, size_t rows, size_t cols )
{
  size_t i, j;

  for (i = 0; i < rows; i++)
  {
    for (j = 0; j < cols; j++)
    {
      a[i * rows + j] = some_value();
    }
  }
}

int main( void )
{
  int arr[10][20];

  foo( &arr[0][0], 10, 20 );
  ...
  return 0;
}

这将适用于堆栈上分配的数组:

T a[M][N];

对于以下形式的动态分配数组:

T (*ap)[N] = malloc( M * sizeof *ap );

因为两者都将具有连续分配的行。对于以下形式的动态分配数组,这将不起作用(或至少不能保证起作用):

T **ap = malloc( M * sizeof *ap );
if (ap)
{
  size_t i;
  for (i = 0; i < M; i++)
  {
    ap[i] = malloc( N * sizeof *ap[i] );
  }
}

因为不能保证所有行都将连续分配给彼此。

于 2013-03-06T11:58:41.980 回答
0

John Bode 的解释很好,但是有个小错误:应该是

i * cols + j

代替

i * rows + j
于 2014-01-24T18:49:41.583 回答
0

这是对@John Bode 的好回答的一种评论

对于以下形式的动态分配数组,这将不起作用(或至少不能保证起作用):

但是这个变体将:

T **ap = malloc( M * sizeof *ap );
if (ap) return NULL;     ---> some error atention
if (ap)
{
  ap[0] = malloc( M * N * sizeof *ap[i] );
  if (ap[0]) { free(ap); return NULL;}     ---> some error atention
  size_t i;
  for (i = 1; i < M; i++)
  {
    ap[i] = ap[0] + i * N;
  }
}

使用后:

free(ap[0]);
free(ap);

因为你T为数组int调用foo了 alsint ap[M][N];

  foo( &ap[0][0], M, N);

因为您保证所有行都彼此连续分配。这种分配效率更高。

于 2013-03-06T13:20:54.803 回答
-1

如果你真的想要引用,那么它只在 C++ 中。

通过引用传递的二维 int 数组示例

void function_taking_an_array(int**& multi_dim_array);

但是参考没有任何优势,所以只需使用:

void function_taking_an_array(int** multi_dim_array);

我建议您使用容器来保存您的阵列。

于 2013-03-06T10:55:44.513 回答