天天看点

c指针解读之左右法则(2)1:int* (a[5])(int, char);2:void (b[10]) (void ()());3:int(*)() (*c)[9];4:int ((*d)[5])(int );5:int ((*e)(int ))[5];

1:int* (a[5])(int, char);

  首先看到标识符名a,”[]”优先级大于”“,a与”[5]”先结合。所以a是一个数组,这个数组有5个元素,每一个元素都是一个指针,指针指向”int (int, char*)”,很明显,指向的是一个函数,这个函数参数是”int, char*”,返回值是”int*”。OK,结束了一个。:)

2:void (b[10]) (void ()());

  b是一个数组,这个数组有10个元素,每一个元素都是一个指针,指针指向一个函数,函数参数是”void (*)()”【注:这也是一个函数指针, 参数为空,返回为void】,返回值是”void”。完毕!

3:int(*)() (*c)[9];

   c是一个指针,指针指向一个数组,这个数组有9个元素,每一个元素都是”int(*)()”(也即一个函数指针,指向一个函数,这个函数的参数为空,返回值是int型)。

4:int ((*d)[5])(int );

(*d)——指针;

(*d)[5]——这个指针指向一个数组;

*(*d)[5]——这个数组中每个元素都是指针类型;

int (int *)—— 什么类型的指针?这个类型的。

5:int ((*e)(int ))[5];

e—–向右遇到括号,向左遇到,说明e是个指针,啥指针呢?

(e)(int )——跳出括号向右遇到(int ),说明这个指针是个函数指针,形参为int, 返回值为何?且听下回分解:);

(*e)(int )——返回值为何?向右遇到括号,再向左,喔,遇到*了,那就是返回了一个指针了。啥指针呢? 同样地,下回分解;

((*e)(int ))[5]——-向右遇到[],说明那是个指向数组的指针,是啥数组呢?不急,慢慢来;

int ((*e)(int ))[5]——-向左遇到int,喔,明白了,就是个简单的整型数组。OVER

当然实际当中,当需要声明一个复杂指针时,如果把整个声明写成上面所示的形式,对程序可读性将是一个巨大损害。谁要是写出这样BT的指针声明,那就真是丢rp了,估计会被骂死!。

还是用typedef来对声明逐层分解替换下吧,增强可读性。

例如对于上面的声明:int ((*func)(int ))[5]; 可以这样分解:

typedef int (*pArr)[5];

typedef pArr (func)(int );

这样就容易读得多了啊!

再看看这个啥意思? typedef int (* (* (FUNC)(int ) )[5] )(int *); —- 晕了吧。

其实typedef int (* (* (FUNC)(int ) )[5] )(int *);

等价与下面的:)

typedef int (PF)(int );

typedef PF (*PARRAY)[5];

typedef PARRAY (FUNC)(int );

((void ()())0)();——->这个呢?

按左右法则:

(void (*)()) —–是一个返回值为void,参数为空的函数指针原型。

  (void (*)())0—–把0强转成一个返回值为void,参数为空的函数指针,指针指向的地址为0.

  (void ()())0—–前面加上*表示整个是一个返回值为void的函数的名字

  ((void ()())0)()——这当然就是一个函数调用了。

再typedef化简下:

typedef void (*pf)();

(*(pf)0)();

继续阅读