C语言 - 数组做函数参数发生了什么?

本文深入探讨了数组作为函数参数时如何退化为指针,详细解释了不同函数声明形式下的行为差异,并提供了正确传递数组和数组长度的示例,避免常见错误。

结论:数组做函数参数时退化为一个指针变量,当我们需要传数组的时候,正确的做法是把数组和数组的有效长度传给函数。

数组做函数参数的几种形式:
一:
void printArray( int arr[] ) {}
二:
void printArray( int arr[10] ) {}
三:
void printArray( int *arr ) {}

为什么说数组做函数参数会退化成一个指针变量?有什么依据?

我们先声明一个数组 :
int arr[] = {1,2,3,4,5,6,7,8,9,10};

然后对数组求大小 :
sizeof(arr) : 40;

这个结果是显而易见的,大小是40字节,10个整型数据类型的大小。

但是,我们把实参arr传递给printArray函数的形参arr以后,再在printArray函数里面对形参arr求大小,包括以上三种函数声明形式:
sizeof(arr[]) : 4;
sizeof(arr[10]) : 4;
sizeof(*arr) : 4;

它们的大小都是4个字节,指针在32位机下也是四个字节。由此可见,数组做函数参数实际上就是把数组的内存的首地址传递给形参,形参是一个指向数组首地址的指针。

所以,初学者喜欢犯的错误就是把数组传递给形参以后,在函数内用sizeof()的方法去求数组的实际大小,这么做求得的大小永远是一个整型数据的大小,所以是错误的。

错误示例:
void fun(int arr / arr[] / / arr[10]*/ ) {
for ( int i = 0; i < sizeof(arr)/sizeof(int); ++i) {

}
}

正确示例:
void fun(int arr / arr[] / / arr[10]*/ , int size ) {
for ( int i = 0; i < size; ++i) {

}
}

<think>嗯,用户想了解如何在C语言中将数组作为函数的输入参数。首先,我需要回顾一下C语言数组作为函数参数的基本用法。根据引用[1]和[3],数组作为参数传递时,实际上传递的是数组的首地址,也就是指针。所以函数内部接收的是一个指针,而不是整个数组的副本。这意味着在函数中对数组元素的修改会影响原数组,这和普通的值传递不同。 接下来,我应该解释三种常见的传参方式:基本形式、带长度的形式,以及二维数组的情况。基本形式的话,函数参数可以写成`int arr[]`或者`int *arr`,这两种写法在编译器看来是一样的。比如引用[1]中的例子,`void func(int arr[])`,这里arr实际上指针。需要提醒用户,这种形式下无法通过sizeof在函数内正确获取数组长度,必须显式传递长度参数,如引用[3]中的GetData和Sort函数都用了参数k来传递元素个数。 然后是二维数组作为参数的情况。这里需要明确,二维数组的列数必须指定,例如`int a[][3]`。这样编译器才能正确计算内存地址。引用[3]中的例子虽然是一维的,但可以扩展到二维的情况,比如传递二维数组时,函数参数需要指定列数。 还要注意指针数组名的区别。虽然数组名在大多数情况下会退化为指针,但在函数参数中,声明为数组的形式实际上还是指针。用户可能会混淆这一点,所以需要明确说明。比如,在函数内部使用sizeof(arr)得到的只是指针的大小,而不是数组的总大小,因此必须传递长度参数。 另外,用户可能需要知道如何避免数组越界,所以在示例中展示如何传递长度参数是必要的。引用[3]中的Sort函数传递了k作为元素个数,这样循环时就不会越界。 最后,生成相关问题的时候,要考虑用户可能的后续疑问,比如指针数组的区别、二维数组传参的注意事项,以及如何防止越界访问。这样可以帮助用户更全面地理解数组作为函数参数的使用。</think>### C语言数组作为函数参数的用法解析 在C语言中,数组作为函数参数时,**传递的是数组首元素地址**而非整个数组。这意味着函数内部可以直接操作原数组数据,且参数传递方式与指针高度关联[^1][^3]。以下是具体用法和注意事项: --- #### 一、一维数组作为参数 **1. 基本形式** 函数定义可接受两种等效形式: - 数组形式:`void func(int arr[])` - 指针形式:`void func(int *arr)` **示例**(引用自[^1]): ```c void func(int arr[]) { // 等效于 int *arr for(int i=0; i<5; i++) { printf("arr[%d]: %d\n", i, arr[i]); } } int main() { int arr[5] = {1,2,3,4,5}; func(arr); // 直接传递数组名 return 0; } ``` **2. 带长度的形式** 需显式传递数组长度,避免越界访问: ```c void Sort(int a[], int k) { // k表示元素个数 for(int i=0; i<k-1; i++) { // 通过k控制循环范围[^3] // 排序逻辑 } } ``` --- #### 二、二维数组作为参数 必须指定列数,且参数声明需明确维度: ```c void processMatrix(int a[][3], int rows) { // 列数必须指定为3 for(int i=0; i<rows; i++) { for(int j=0; j<3; j++) { a[i][j] *= 2; } } } ``` --- #### 三、关键注意事项 1. **数组长度丢失问题** 函数内无法通过`sizeof(arr)/sizeof(arr[0])`获取数组长度,必须显式传递长度参数[^2][^3]。 2. **指针数组名的区别** 虽然`arr`和`&arr[0]`值相同,但`&arr`表示整个数组的地址(类型为`int(*)[N]`)。 3. **动态数组传参** 动态分配的数组(如`int *arr = malloc(5*sizeof(int))`)传参方式与静态数组一致。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值