指针数组与数组指针:
指针数组:是由指针组成的数组,它的成员都是指针变量
类型* arr[长度];
数组指针:专门指向数组的指针
类型 (*arrp)[长度];
指针与数组名:
数组名是一种特殊的指针,它是一个常量,不能修改它的值,它与数组的内存是映射关系,
没有自己的内存空间。
&数组名 == 数组名 == &数组名[0]
指针变量它有自己的存储空间,如果它储存的是数组的首地址,指针可以当做数组使用,
数组名也可做指针使用。 (指针与数组名用法互通)
数组名[i] == *(数组名+i)
*(指针名+i) == 指针名[i]
数组作为函数的参数时蜕变成了指针,所以长度丢失。
区别:数组名与数组内存是映射关系,指向数组内存的指针与数组内存是指向关系。
二级指针:
二级指针就是指向指针的指针。里面存储的指针变量的地址。
定义:
类型** 变量名_pp;
赋值:
变量名_pp = 地址;
<=> 变量名_pp = &指针变量; (&p)
解引用:
*变量名_pp = 指针变量 (p)
**变量名_pp = *指针变量 (*p) == 普通变量
注意:当函数之间需要共享指针变量时,必须使用二级指针传递
函数指针:
函数返回值(*函数指针名p)(参数列表);
函数名就是一个地址(整数),它代表了函数在代码段中的位置。
可以通过函数指针,把函数作为参数传递给另一个函数,这叫回调。
函数指针就是指向函数的指针,里面存储的就是该函数在代码段的位置。
int scanf(const char *format,...);
函数名其实就是函数在代码段中的内存的首地址。
调用函数其实就是跳转到函数所在的代码段中的位置,然后执行二进制指令。
函数指针中储存了函数的首地址,对它解引用就可以执行函数
定义函数指针:
返回值类型(*指针变量名)(类型1,类型2,...);
赋值:
指针变量名 = 函数名;
解引用:
指针变量名(实参);