在学习的过程中,把学到的知识写出来一篇博客,可以帮助我们融会贯通,对知识更了解,记忆更深刻。
1.数组指针变量
首先要分清指针数组和数组指针,指针数组是一个数组里边的元素是指针,数组指针是一个指针指向数组。
整形指针--变量里存放的是整形的指针(地址)
字符指针--变量里存放的是字符的指针(地址)
数组指针--变量里存放的是数组的指针(地址)。
前边我们讲过&数组名,取到的就是整个数组的地址。
数组指针的写法:int(*p)[10]=&arr,当然这个p和10都是可以变得,具体看实际需求。因为如果写成int*p[10],那么p就会先和[10]结合,那这样就变成指针数组了,所以要把*p括到一起,让他们两个先结合,表示p是一个指针,然后指向数组,数组里边有10个元素,每个元素是int类型。
int(*p)[10]=&arr,这里的p就是数组指针变量名。这里的int是arr数组里元素的类型,[10]是数组的元素个数。
2.二维数组传参的本质
这是我们正常传参二维数组,打印二维数组的方法。
我们发现,二维数组传参传递的也是数组名,而数组名就是数组首元素的地址,那既然实参传过去的是数组首元素的地址,那形参也可以用指针变量来接收。
那二维数组的数组名,到底表示的是谁的地址?首元素是谁?
首先我们要知道二维数组其实就是一维数组的数组,二维数组的每个元素是一维数组。
所以,二维数组的首元素应该就是他的第一行。二维数组的数组名表示的就是首元素的地址,那么就是第一行的地址。
那既然我们传过去的是第一行的地址,那么写成指针的方式,就可以写成数组指针。
就是这样。这个arr是指向一行的,是整个一维数组的地址,那arr+1就代表第二个一维数组,也就是第二行的地址,同理arr+2就代表第三行的地址。那我们就可以把代码改造成这样:
这个要怎么理解呢? 首先*(arr+i)这里我们拿到的是每一行首元素的地址,然后让他们+j,就拿到跟这个首元素一行之后元素的地址,然后再对(*(arr+i)+j)解引用,就得到每个地址的元素本身。
这个*(arr+i)可以写成p[i],同理*(*(arr+i)+j)就可以写成p[i][j]。
3.函数指针变量
既然变量有地址,数组有地址,那么函数有地址吗?能取地址吗?其实是可以的。
函数也是有地址的,并且&add和add都是函数的地址。&函数名和函数名都可以取到函数的地址。
函数指针变量和数组指针变量的写法很相似,*p表示p是一个指针,(int,int)表示p指向的是一个函数,而这个函数的返回类型是int,所以在前边加上int,而这个函数的地址&add,它的类型就是int(*)(int,int)。这里的p就是函数指针变量
那使用它,还是解引用,然后传参。
在调用的时候,(*p)(2,3)其实可以不写,p(2,3)也是可以的,这是语法层面的一个设计,为了使用起来更方便。
如有助益,吾之所幸。