关键概念:
1.多个不同类型的指针可以对应同一个地址;
2.(&p)则是这样一种运算,返回一个指针,该指针的值是当时声明p 时开辟的地址,指针的类型是p的类型对应的指针类型;
3.(*p)操作是这样一种运算,获取指针p的值(变量的地址)找到对应的变量的值-----返回以(p 的值作为地址)的那个空间的值。但是如果这个变量本身就是一个地址,那么(*p)获取的值也是一个指针;
4.在数组中,指针的&,*运算符比较特殊,与普通变量的指针并不相同;
5.数组名不是一个该数组的指针;
一维数组
#include<stdio.h> #include<stdlib.h> void main(){ int arr[5] = {1,2,3,4,5}; //通过int *p = arr这个定义,首先要明确arr的类型是int * int *p = arr; //c语言定义:当一个数组标识符出现在表达式中, //①这个标识符的类型就从“某种类型t的数组”转换成“指向类型t的指针”, //而且它的值就等于数组第一个元素的地址。并且这个指针是个常量指针,不可改变其值的。 //以上的叙述很官方,但是大家对①这个描述并不是很清楚,我以一维数组arr为例来解释一下 //“某种类型t的数组”指的就是数组arr[5],“指向类型t的指针”指的是数组第一个元素arr[0], //通俗解释就是这样子的,②当一个arr出现在表达式中,arr所代表的就是数组第一个元素的指针, //②同样适用于多维数组,这个具体讲解将会在二维数组中详细介绍。 //在本例子中,arr就是arr[0]的指针。 //有了以上的基础,我们来讨论一下arr+1,&arr+1 //打印出数组arr中所有的元素和他们的地址 for (int i = 0; i < 5; i++) { printf("%5d--%7x",arr[i],&arr[i]); } printf(" "); //下面我们来分析arr+1的值,通过②,我们可以知道arr的类型是整型指针, //假设arr的地址是0x87f904,那么arr+1的地址就是在arr的值(指针的值是一个地址,以前文章有说明)上加上4个字节(char型1字节,int型4字节) //所以根据分析arr+1的地址应该是0x87f908 printf(" arr=%x,arr + 1=%x", arr, arr + 1); //以上代码的结果证明我的分析是正确的 //现在对关键概念4做第一个证明, //对&arr做一个解释 printf(" arr=%x,&arr=%x", arr, &arr); //通过以上代码,我们发现arr和&arr的地址完全相同, //但是实际上&arr(为什么说&arr是一个指针,请参照关键概念2)的指针类型和arr的指针类型却完全不同 //&arr返回的是arr数组的指针,&arr指针的类型是int(*)[5], //为此我们可以通过sizeof()运算符来查看&arr指针的类型 //sizeof(*(&arr)),在这里我要解释一下,为什么不是sizeof(&arr) //根据关键概念2我们可以知道&arr本身是一个指针,sizeof(指针),那么大小一定是4个字节 //我们要看一个指针的类型,还可以通过指针对应变量的类型来判断 //所以用*(&arr)获取&arr这个指针对应的变量 //*(&arr)又可以写成arr,所以sizeof(*(&arr))=sizeof(arr) //所以有些资料上会这么说“sizeof(数组名)返回的是整个数组的大小”, printf(" sizeof(*(&arr))=%d,sizeof(arr)=%d", sizeof(*(&arr)), sizeof(arr)); //结果显示sizeof(*(&arr))=20,正好是一个数组arr的大小 //理解完&arr的本质后,我们在对&arr做加减运算就容易理解了,&arr+1,那么指针移动的单位是20个字节 printf(" &arr+1的地址=%x", &arr+1); //以上代码证明我的结论 printf(" "); system("pause"); }
运行结果截图:
二维数组
#include<stdio.h> #include<stdlib.h> void main(){ //对于二位数组,我们主要需要理解arr,*arr,&arr三个指针的区别 int arr[3][4] = {0}; //为二维数组赋值 for (int i = 0; i < 3; i++) { for (int j = 0; j < 4; j++) { printf("%5d,%7x", arr[i][j] = i * 2 + j, &arr[i][j]); } printf(" "); } //判断一个指针的类型,可以通过*运算符求指针对应变量的大小来判断 printf(" *arr==%d", sizeof(*(*arr))); //打印 4 printf(" &arr==%d", sizeof(*(&arr))); //打印 48 printf(" arr==%d", sizeof(*arr)); //打印 16 //由以上结果推断*arr是整型指针,&arr是二维数组指针,arr是一维数组指针(即行指针) printf(" %d", *(*arr));//打印 0 //以上结果表明*arr是a[0][0]的指针,说明其*arr是一个列指针 //arr[0][1]=*(*arr+1) //因为arr是一维数组指针,那么a[1][0]的指针是*(arr+1) //a[1][0]的指针是*(arr+1)+1 //a[1][0]=*(*(arr+1)+1); //再次,我们要更深入一步,用指针来获取二维数组的值 int x = 1, y = 2; //如何获取a[x][y]的值 //通俗的说法,就是第几行的第几个的值 //首先行的地址指针是arr+x,这里arr+x是一个行指针,但是我们获取的最终的值其实是列指针的值, //*(arr+x)这就是arr[x][0]这个整型变量的指针 //y=2说明列指针向后移动2个单位(8个字节),所以*(arr+x)+2就是arr[x][y]的指针 //求指针对应变量的值,用*运算符, //最终结果是*(*(arr+x)+y)==arr[x][y] printf(" *(*(arr+x)+y)=%d,arr[x][y]=%d", *(*(arr + x) + y), arr[x][y]); //因为*(arr + x)可以简写成arr[y] //所以*(*(arr + x) + y)=*(arr[y] + y) printf(" "); system("pause"); }