最近刚开始刷剑指offer,刚做到第三题的时候,发现C++二维数组的传参方式和C语言略有些不同,所以在这篇博客中,会列出C/C++常见的二维数组传参方式。(本方式和代码都是基于vs环境所编写)
一.C语言二维数组传参方式
C语言二维数组传参一般有三个方式。
1.指针形式接收
#include<stdio.h>
void Print1(int* parr, int rows, int cols)//指针,行数,列数
{
for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
printf("%2d ", *(parr + i * cols + j));
}
printf("\n");
}
printf("\n");
}
int main()
{
int arr[][5] = { {1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15},{16,17,18,19,20} };
Print1(arr, sizeof(arr) / sizeof(arr[0]), sizeof(arr[0]) / sizeof(arr[0][0]));
return 0;
}
2. 数组形式接收
#include<stdio.h>
void Print2(int parr[][5], int rows, int cols)
{
for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
printf("%2d ", parr[i][j]);
}
printf("\n");
}
printf("\n");
}
int main()
{
int arr[][5] = { {1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15},{16,17,18,19,20} };
Print2(arr, sizeof(arr) / sizeof(arr[0]), sizeof(arr[0]) / sizeof(arr[0][0]));
return 0;
}
3.数组指针形式接收
void Print3(int(*arr)[5], int rows, int cols)
{
for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
printf("%2d ", (*(arr + i))[j]);
}
printf("\n");
}
printf("\n");
}
int main()
{
int arr[][5] = { {1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15},{16,17,18,19,20} };
Print3(arr, sizeof(arr) / sizeof(arr[0]), sizeof(arr[0]) / sizeof(arr[0][0]));
return 0;
}
一般来说,第一种更为常见,更加通用,因为第二、三种方法需要在函数中写明二维数组的列数,而一般情况下是不知道的。
二.C++二维数组传参方式
当我写完C语言二维输出传参方式后,接着用C语言写,发现C语言的第一种写法不适用于C++,会直接报错。
原因:
在C语言里面,不管是一维数组还是二维数组,通过用数组名的传参方式,传过去后,数组名都会退化成一个指针指向第一个元素的指针。
而在C++的二维数组里面,通过用数组名传参,传过去后数组名会退化成一个一维数组指针,即这个指针是指向一个一维数组的。
如下图所示:
所以C++的函数参数不能像C语言一样去写。
1. 数组指针形式接收
#include<iostream>
using namespace std;
void Print1(int (*parr)[5], int rows, int cols)
{
for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
cout << (*(parr + i))[j] << " ";
}
cout << endl;
}
cout << endl;
}
int main()
{
int arr[][5] = { {1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15},{16,17,18,19,20} };
Print1(arr, sizeof(arr) / sizeof(arr[0]), sizeof(arr[0]) / sizeof(arr[0][0]));
return 0;
}
2.指针形式接收
第一种方式参数要指明多少列,如果我们不想这样写可以吗,答案是可以的,我们还是可以像C语言的第一种写法一样去写,但是调用函数时的参数需要做出一些改变。
传arr改为传arr[0][0]的地址。
#include<iostream>
using namespace std;
void Print2(int* parr, int rows, int cols)
{
for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
cout << *(parr + i * cols + j) << " ";
}
cout << endl;
}
cout << endl;
}
int main()
{
int arr[][5] = { {1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15},{16,17,18,19,20} };
//函数第一个参数改为传arr[0][0]的地址
Print2(&arr[0][0], sizeof(arr) / sizeof(arr[0]), sizeof(arr[0]) / sizeof(arr[0][0]));
return 0;
}
3. 引用方式传参
void Print3(int(&parr)[4][5],int rows,int cols)
{
for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
cout << parr[i][j] << " ";
}
cout << endl;
}
cout << endl;
}
int main()
{
int arr[][5] = { {1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15},{16,17,18,19,20} };
Print3(arr, sizeof(arr) / sizeof(arr[0]), sizeof(arr[0]) / sizeof(arr[0][0]));
return 0;
}
这种引用方式传参要指定行和列,所以也不好用,但是好在C++支持模板,所以可以使用模板来优化。
4.引用+模板
template<size_t row, size_t col>
void Print4(int(&parr)[row][col])
{
for (int i = 0; i < row; i++)
{
for (int j = 0; j < col; j++)
{
cout << parr[i][j] << " ";
}
cout << endl;
}
cout << endl;
}
int main()
{
int arr[][5] = { {1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15},{16,17,18,19,20} };
Print4(arr);
return 0;
}
通过使用引用+模板的方式传参,通过使用模板,编译器会自动推导数组的行数和列数。