目录
13.指针的使用和传址调用
14.数组名的理解
*数组名就是数组首元素的地址
*两个例外
*使用指针访问数组
*一维数组的传参本质
往期推荐
承接上篇39.【C语言】指针(重难点)(D)
13. 指针的使用和传址调用
见29.【C语言】函数系列中 自定义函数 中的1.自定义详解和2.疑问解答
总结:
传址调用,可以让函数和主调函数之间建立真正的联系,在函数内部可以修改主调函数中的变量;传值调用,只是需要主调函数(main函数)中的变量值来实现计算
14.数组名的理解
*数组名就是数组首元素的地址
#include <stdio.h>
int main()
{
int arr[3] = { 0 };
printf("%p\n", &arr[0]);
printf("%p\n", &arr);//&arr下面会讲例外
printf("%p\n", arr);
return 0;
}
*注意有两个例外:
1.sizeof(数组名):这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节
详细见20.5.【C语言】求长度(sizeof和strlen)的两种方式
2.&数组名,这里的数组名表示整个数组,取出的是整个数组的地址(不等同于数组首元素的地址)
尽管上方三个打印的结果是一样的,但只要略微改动
#include <stdio.h>
int main()
{
int arr[3] = { 0 };
printf("%p\n", (& arr[0]) + 1);
printf("%p\n", (& arr) + 1);
printf("%p\n", arr+1);
return 0;
}
C4-BC==8
显然&arr+1的+1 操作是跳过整个数组
*使用指针访问一维数组
#include <stdio.h>
int main()
{
int arr[] = { 0,1,2,3,6,3,9,5,2 };
//计算数组的长度
size_t sz = sizeof(arr) / sizeof(arr[0]);
int* pi = &arr[0];
for (int i = 0; i < sz; i++)
{
printf("%d ", *(pi + i));
}
return 0;
}
注意:*(pi+i)不能写成*pi+i!运算顺序不一样!
由于arr[i]编译器在执行时会转换为*(arr+i),且p == arr
一个大胆的猜想:arr[i]==i[arr],执行后结果正确(无论哪种写法都会转换为*(arr+i),[]只是操作符)
则有*(pi+i)==*(i+pi)==*(arr+i)==*(i+arr)==arr[i]==i[arr]==p[i]==i[p]
int*中*(pi+1)跳过4个字节
*一维数组的传参本质
#include <stdio.h>
void test(int arr[10])
{
int sz2 = sizeof(arr) / sizeof(arr[0]);
printf("sz2 = %d\n", sz2);
}
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
int sz1 = sizeof(arr) / sizeof(arr[0]);
printf("sz1 = %d\n", sz1);
test(arr);
return 0;
}
(左边是x64,右边是x86)
所以传参的时候并没有传递整个数组
以x86环境为例说明:sz2=sizeof(arr)/sizeof(arr[0]),sz2==1,即sizeof(arr)==1,回想:数组名是数组首元素的地址,那么在数组传参的时候,传递的是数组名
总结:1.数组传参本质上是传递的是数组首元素的地址
2.函数形参部分不会真实创建数组,那么就不需要数组的大大小
3.函数形参部分应该用指针变量接收 int* p
4.对于一维数组,形参既可以写成数组的方式,也可以写成指针变量的方式
以下写法均可以
void test(int arr[10])
void test(int arr[])
void test(int* arr[10])
void test(int* arr[])
往期推荐
19.【C语言】指针(重难点)(A)
37.【C语言】指针(重难点)(B)
38.【C语言】指针(重难点)(C)
39.【C语言】指针(重难点)(D)