理解复杂声明可用的“右左法则”:从变量名看起,先往右,再往左,碰到一个圆括号就调转阅读的方向;括号内分析完就跳出括号,还是按先右后左的顺序,如此循环,直到整个声明分析完。举例:
int (*func)(int *p);
首先找到变量名func,外面有一对圆括号,而且左边是一个*号,这说明func是一个指针;然后跳出这个圆括号,先看右边,又遇到圆括号,这说明(func)是一个函数,所以func是一个指向这类函数的指针,即函数指针,这类函数具有int类型的形参,返回值类型是int。
int (*func[5])(int *);
func右边是一个[]运算符,说明func是具有5个元素的数组;func的左边有一个*,说明func的元素是指针(注意这里的不是修饰func,而是修饰func[5]的,原因是[]运算符优先级比高,func先跟[]结合)。跳出这个括号,看右边,又遇到圆括号,说明func数组的元素是函数类型的指针,它指向的函数具有int*类型的形参,返回值类型为int。
void (*func[10]) (void (*)());
func右边是一个[]运算符,说明func是具有10个元素的数组;func的左边有一个*,说明func的元素是指针,跳出括号,看右边,说明数组元素是void(*) ((void(*)())
类型的函数指针,指向的函数具有void(*)()类型的形参,返回值为void
简化代码可使用typedef
typedef void (*pFunParam)();\\先简化右边的
typedef void (*pFunx)(pFunParam);\\再简化左边的
pFunx b[10];
int (a[5])(int, char);
a是五个元素的指针数组
元素是int(*)(int,char*)
类型的函数指针
指向的函数返回值为int*
具有(int,char*)
类型的形参
double()() (pa)[9];
pa是九个元素的指针数组
元素是double(*)()
类型的函数指针
int * ( (*fp1) (int) ) [10];
fp1是一个函数指针
指向一个参数为int,返回值为一个指针的函数
这个函数的返回值指向一个int*
类型的指针数组
虽然比较少用,但是我个人觉得还是有必要掌握的。
因为既然这样的问题今后会碰到,那么花一次大的力气把它掌握以后就可以不用再费脑筋猜这是什么含义了。但是要指出的是工作中很多公司都禁止写2层以上的指针,所以几乎不会碰到,也算是无聊的东西了。