接上一期我们讲到了野指针的概念,野指针的成因和避免野指针的方法,又讲了指针的运算,今天我们接着讲指针新的内容,指针的每一章都很重要,今天也不要忘记学习。
1.指针和数组
我们在调用函数时,需要用到数组时,例如我们创建了应该arr【10】的数组,传入的都是不带任何下标的arr本身,那么传入arr表示我们传入了arr数组中的所有元素吗,并不是,传入arr表示传入了arr数组的首地址,即arr=&arr【0】,代码演示:
#include<stdio.h>
#include <stdio.h>
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,0};
printf("%p\n", arr);
printf("%p\n", &arr[0]);
return 0;
}
我们分别打印arr与&arr【0】的地址,如果它们的地址相等,就意味着 arr=&arr【0】,也就是当传入arr时,就是传入了arr数组首元素的地址
运行程序,我们可以很直观地看到arr与&arr【0】的地址时一样的。
总结:数组名就是数组首元素的地址 。
那么我们写出这样的代码是可行的:
int arr[10] = {1,2,3,4,5,6,7,8,9,0};
int *p = arr;//p存放的是arr数组首元素的地址
如果可以把数组名当成地址放入,那么我们也可以用指针来访问数组,进而遍历整个数组
例如:
#include <stdio.h>
int main()
{
int arr[] = {1,2,3,4,5,6,7,8,9,0};
int *p = arr; //指针存放数组首元素的地址
int sz = sizeof(arr)/sizeof(arr[0]);
for(i=0; i<sz; i++)
{
printf("&arr[%d] = %p <====> p+%d = %p\n", i, &arr[i], i, p+i);
}
return 0;
}
这次我们分别用数组下标和指针来访问数组,看看结果吧:
毫无疑问,我们是可以通过把数组首地址给指针去遍历整个数组的,所以 p+i 其实计算的是数组 arr 下标为i的地址。
int main()
{
int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 };
int *p = arr; //指针存放数组首元素的地址
int sz = sizeof(arr) / sizeof(arr[0]);
int i = 0;
for (i = 0; i<sz; i++)
{
printf("%d ", *(p + i));
}
return 0;
}
这次我们完全使用指针来访问数组:
事实证明这种方法是完全可行的 。
2.二级指针
二级指针是存放一级指针地址的指针,什么意思呢,指针变量也是变量,是变量就有地址,我们经常见到的指针类似int* p被称为一级指针,而指针的地址存放并不是也存放在和它同等级的一级指针内的,所以,高级指针没有那么复杂,高级指针的作用就是用来存放比它本身低一级指针的地址的。
例如:
int main()
{
int a = 10;
int* pa = &a;
int** ppa = &pa;
printf("%p\n", &a);
printf("%p\n", pa);
printf("%p\n", &pa);
printf("%p\n", ppa);
printf("%p\n", &ppa);
return 0;
}
运行这段代码:
接下来看图理解吧:
3.指针数组
指针数组,它是指针还是数组呢,答案是数组,它是包含了若干个指针变量的数组。
char arr[8];
这行代码怎么表示呢,看图理解:字符数组
int arr2[8];
那么指针数组是怎样的呢?
int* arr3[6]
arr3是一个数组,有六个元素,每个元素是一个整形指针。
总结:指针数组是一个存放若干个指针变量的数组。
这一期的内容到这就结束了,我们下期不见不散 。
未完待续。。。。。。