一、数组名的理解
1、数组名代表首元素地址
#include<stdio.h>
int main()
{
int arr[5] = { 1,2,3,4,5 };
printf("arr = %p\n", arr);
printf("&arr[0] = %p\n", &arr[0]);
return 0;
}
数组名和数组第一个元素的地址相同
2、数组名有两个例外
1、sizeof(数组名)
#include<stdio.h>
int main()
{
int arr[5] = { 1,2,3,4,5 };
printf("%zu\n", sizeof(arr));
return 0;
}
sizeof()里面的数组名代表整个数组
2、&数组名
取地址数组名
#include<stdio.h>
int main()
{
int arr[5] = { 1,2,3,4,5 };
printf("arr = %p\n", arr);
printf("&arr[0] = %p\n", &arr[0]);
printf("&arr = %p\n", &arr);
printf("arr+1 = %p\n", arr+1);
printf("&arr[0]+1 = %p\n", &arr[0]+1);
printf("&arr+1 = %p\n", &arr+1);
return 0;
}
可以发现&数组名+1跳过整个数组,所以&数组名代表整个数组的地址
二、使用指针访问数组
指针指向第一个元素的地址,可以通过指针来打印数组中的每个元素
#include<stdio.h>
int main()
{
int a[5] = { 1,2,3,4,5 };
int* p = a;
for (int i = 0; i < 5; i++)
{
printf("%d ", *(p + i));
}
return 0;
}
当指针获得数组的第一个元素的地址,通过指针的加减运算得以得到每个元素的值,这是应为数组在内存中是连续存放的
三、一维数组传参的本质
在函数调用中一维数组把实参传个形参时,形参可以写成数组的形式,也可以写成指针的形式,形参的改变会影响实参的本质是,形参拿到的是实参的地址,是一次传址调用。
#include<stdio.h>
void test(int a[], int sz)
{
for (int i = 0; i < sz; i++)
{
scanf("%d", &a[i]);
}
}
int main()
{
int a[5] = { 1,2,3,4,5 };
int sz = sizeof(a) / sizeof(a[0]);
test(a,sz);
for (int i = 0; i < sz; i++)
{
printf("%d ",a[i]);
}
return 0;
}
通过函数调用实现改变实参中的值,形参部分写成指针的形式
写出指针的形式如下:
#include<stdio.h>
void test(int *a, int sz)
{
for (int i = 0; i < sz; i++)
{
scanf("%d", &a[i]);
}
}
int main()
{
int a[5] = { 1,2,3,4,5 };
int sz = sizeof(a) / sizeof(a[0]);
test(a,sz);
for (int i = 0; i < sz; i++)
{
printf("%d ",a[i]);
}
return 0;
}
其实a[i] = *(a+i)
四、二级指针
指针变量存放变量的地址,用二级指针就可以存放指针变量的地址。
#include<stdio.h>
int main()
{
int a = 20;
int* p = &a;
printf("a = %d\n", a);
printf("*p = %d\n", *p);
printf("&a = %p\n", &a);
printf("p = %p\n", p);
int** q= &p;
printf("*q = %p\n", *q);
printf("**q = %d\n", **q);
return 0;
}
一级指针p,p里面存放的是&a
*p,解引用p就等于a
&p,是指针p的地址
用二级指针q来存放,q存放的是&p
*q,解引用就等于p,p就等于&a
**q,解引用*q,就等于解引用p,就等于*&a,等于a的值
五、指针数组
指针数组本质上是数组,数组里面的元素是指针类型的
#include<stdio.h>
int main()
{
int a[5] = { 1,2,3,4,5 };
int* p[5] = { &a[0],&a[1],&a[2],&a[3],&a[4] };
for (int i = 0; i < 5; i++)
{
printf("%d ", **(p + i));
}
return 0;
}