备用知识
指针数组: int *p[10];
数组指针: int (*p)[10];
函数声明: int* f( ); //f是一个返回int*指针的函数
函数指针: int (*p)( );【突破口】
——当前变量是函数?数组?指针?某种类型的强制转换?
学会把解析完的内容看成一个整体!
(1)是函数:明确函数的形参与返回值
(2)是指针数组:明确指针数组的元素的类型
(3)是数组指针:明确指针指向的数组的元素类型
(4)是函数指针:明确函数指针指向的函数的形参与返回值
(5)是某种类型的强制转换:类型进行用括弧()括起来,后边跟着一个其他类型的变量,见特例(* (void(*)())0) ()解读
高级(复杂)指针的含义
解读下面的几个很好的例子
int *(*p)[10]; //p是一个指针(p指向一个数组(该数组的元素类型是int*))
int **(*p[6])(); //p是一个数组(该数组的元素类型是int*的指针(该指针是一个指向形参为void返回值为int**的函数指针))
int *(*p())(); //p是一个函数(该函数形参为void,返回值是int*类型的指针(该指针又是一个指向形参为void,返回值为int*类型的函数指针))
int (**(*p)())(); //p是一个指针(p指针指向形参为void,返回值为int**的二级指针(该二级指针是形参为void,返回值为int类型的函数指针))
int (*(*p)())[6]; //p是一个函数指针(p指向形参是void,返回值是int*类型的指针(该返回的指针指向一个数组(数组元素是int类型)))
char* (*p[3])(char* p);//p是一个数组(数组元素为char*类型的指针(该指针指向形参为char*,返回值为char*类型的函数))
int *(*(*(*p)())[6])() //p是一个函数指针(p函数指针指向形参是void,返回值是int*的指针(该返回的指针指向一个数组(数组类型为int*指针(int*指针又指向形参为void,返回值为int*的函数))))
(* (void(*)())0) ()解读
(*(void(*)())0) ()
//等价于(*(void(*p)())0)()
//void(*p)():p是一个函数指针(指向形参为void,返回值为void的函数),实际上void(*p)()是一种类型,可用作强制转换
//(void(*p)()) 0 :把0强制转换成上述说的void(*p)()类型,该类型是函数指针类型
//*(void(*p)())0 :转换完成后,用*进行间接访问函数指针里边的内容(因为是该指针是函数指针,因此用*访问该指针得到的内容肯定是某个函数的名称,即该函数的地址)--->因此*(void(*p)())0的含义是取出0地址处的函数的地址
//(*(void(*)())0) () :因为上面已经取出了函数的地址,(*(void(*)())0) () 就是函数指针的调用形式。
【扩展】解读(*(char** (*)(char**,char**))0)(char**,char**)
高级(复杂)类型转换符
如何获得某个类型的类型转换符?
①首先声明一个给定类型的变量;
②然后把声明中变量名和声明末尾的分号去掉;
③最后将剩余的部分用一个括号整个“封装”起来即可。
例:
float (*p)();//声明p是一个指向形参为void返回值为float的函数指针
(float (*)()) //表示一个“指向形参为void返回值为float的函数指针”的类型转换符