C/C++二维数组传参

在C/C++中,二维数组传参时需要指定第二维长度。三种常见写法在性能上无差别,均通过指针传递。通过Compiler Explorer可查看汇编代码。main函数示例展示了如何覆盖参数数组值。腾讯云的技术分享提到,利用模板可以方便地复用传参代码,以应对二维数组需指定长度的问题。

C/C++里二维数组传参时需要指定每行有几个元素(第二维长度),具体有三种常见的写法:  

// 参数传int a[][]过不了类型检查
void foo1(int a[][2]) {
    a[1][1] = 10;
}

void foo2(int *a[2]) {
    a[1][1] = 10;
}

void foo3(int **a) {
    a[1][1] = 10;
    // 等价的寻址写法
    // *(*(a+1)+1) = 10;
}

这三种写法在传参的性能上没有区别,在64位机器上都是传了8 bytes的指针,可以在 Compiler Explorer 上在线查看对应的汇编代码:

foo1(int (*) [2]):
        push    rbp
        mov     rbp, rsp
        mov     QWORD PTR [rbp-8], rdi
        mov     rax, QWORD PTR [rbp-8]
        add     rax, 8
        mov     DWORD PTR [rax+4], 10
        nop
        pop     rbp
        ret
foo2(int**):
        push    rbp
        mov     rbp, rsp
        mov     QWORD PTR [rbp-8], rdi
        mov     rax, QWORD PTR [rbp-8]
        add     rax, 8
        mov     rax, QWORD PTR [rax]
        add     rax, 4
        mov     DWORD PTR [rax], 10
        nop
        pop     rbp
        ret
foo3(int**):
        push    rbp
        mov     rbp, rsp
        mov     QWORD PTR [rbp-8], rdi
        mov     rax, QWORD PTR [rbp-8]
        add     rax, 8
        mov     rax, QWORD PTR [rax]
        add     rax, 4
        mov     DWORD PTR [rax], 10
        nop
        pop     rbp
        ret

由于都传的是指针而非值拷贝,这三种写法都可以做到覆盖参数中的数组值,具体可以试试下面的main函数:

#include <cstdio>

// code for foo1,foo2,foo3 omitted

int main() {
    int a[2][2];
    foo1(a);
    // printf("%d\n", a[1][1]);
    printf("%d\n", *(*(a+1)+1));

    int *b[2] = {new int[2], new int[2]};
    foo2(b);
    // printf("%d\n", b[1][1]);
    printf("%d\n", *(*(b+1)+1));
    delete b[0];
    delete b[1];

    int **c;
    c = new int*[2];
    c[0] = new int[2];
    c[1] = new int[2];
    foo3(c);
    // printf("%d\n", c[1][1]);
    printf("%d\n", *(*(c+1)+1));
    delete c[0];
    delete c[1];
    delete c;
}

另外,腾讯云上的一篇 技术分享 提到了基于模版的写法,因为二维数组传参要指定第二维长度,使用模版可以方便复用。

### C++ 中传递二维数组作为函数参数的方法 在C++中,可以通过多种方式将二维数组传递给函数。一种常见的方式是指针表示法。 当定义一个接受二维数组的函数时,可以指定数组的第一维大小为任意值,而第二维度则需固定。这通常通过指针来实现: ```cpp void processArray(int (*array)[COLS], int rows); ``` 上述声明表明 `processArray` 函数接收指向具有 COLS 列数的一维整型数组的指针以及行的数量[^1]。 另一种方法是利用模板编程技术创建能够处理不同尺寸的二维数组的泛化版本: ```cpp template<typename T, std::size_t ROWS, std::size_t COLS> void templateProcess(T (&matrix)[ROWS][COLS]) { // 处理矩阵逻辑... } ``` 此代码片段展示了如何使用模板让编译器自动推导出传入数组的具体行列数目[^2]。 对于动态分配内存的情况或者不知道确切列宽的情形下,还可以采用双重指针的形式,即先申请一块存储空间用于保存每一行首地址的信息,再分别为各行开辟相应长度的空间。不过这种方式较为复杂,在实际应用中应谨慎考虑其必要性和性能影响。 为了简化操作并提高灵活性,现代C++推荐使用标准库容器如std::vector替代传统多维数组结构。例如,可以用嵌套向量(vector of vectors)轻松构建和管理不规则形状的数据集。 ```cpp #include <iostream> #include <vector> // 使用 vector 表达二维数组 void display(const std::vector<std::vector<int>>& matrix){ for(auto& row : matrix){ for(auto elem : row){ std::cout << elem << ' '; } std::cout << '\n'; } } int main(){ std::vector<std::vector<int>> myMatrix{{1, 2}, {3, 4}}; display(myMatrix); return 0; } ``` 这段程序演示了怎样借助 STL 容器类来高效地表达和遍历二维数据集合[^3]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值