首先具备知识:
如何声明一个给定类型的标量,那么该类型的类型转换就很容易得到了:只需要把声明中的变量名 和声明末尾的分号去掉,再将剩余的部分用一个括号整个“ 封装”起来即可。例如:因为下面的声明:
float (*h)();
表示h是一个指向返回值为浮点类型的函数的指针,因此,
(float(*)())
表示一个“指向返回值为浮点类型的函数的指针”的类型转。
(*(void(*)())0)(),这是在C陷阱与缺陷中,关于解决计算机开机启动后,硬件读取首地址为0位置的子例程的代码。
第一步:假定fp是一个函数指针,那么如何调用fp所指向的函数呢?调用方法如下:
(*fp)();
因为fp是一个函数指针,那么*fp就是该指针所指向的函数,所以(*fp)()就是调用该函数的方式。ANSIC标准允许程序员将上式简写为fp(),但是一定要记住这种写法知识一种简写形式。
现在,剩下的问题就只是找到一个恰当的表达式来替换fp。我们将在分析的第二部来解决这个问题。如果C编译器能够理解我们大脑中对于类型的认识,那么我们可以这样写:
(*0)();
上式并不能生效,因为运算符*必须要一个指针来做操作数。而且,这个指针还应该是一个函数指针,这样经运算符*作用后的结果才能作为函数被调用。因此,在上式中必须对0作类型转换,转换后的类型可以大致描述为:“指向返回值为void类型的函数的指针“。
如果fp是一个指向返回值为void类型的函数指针,那么(*fp)()的值为void,fp的声明如下:
void (*fp)();
因此,我们可以用下式来完成调用存储位置为0的子程序:
void (*fp)();
(*fp)();//此处假设fp默认初始化为0,这种写法不宜提倡。
这种写法的代价是多声明了一个“哑”变量。
但是,我们一旦知道如何声明一个变量,也就自然知道如何对一个常数进行类型转换,将其转型为该变量的类型:只需要在变量声明中将变量名去掉即可。因此,将常数0转换为“指向返回值为void的函数的指针”类型,可以这样写:
(void (*)())0
因此,我们可以用(void (*)())0来替代fp,从而得到:(*(void (*)())0)();
末尾的分号使得表达式成为一个语句。
当然,以下方式的书写可以使表达更明确:
typedef void (*func)();//定义了一个指向返回值是void类型的函数指针
(*(func)0)(); //用上面的指针实现强制转换
上面大部分是书上的解释。其实很简单就是先定义一个(*0)();这么一个函数,因为C语言不认识,所以需要进行强制类型转换,转换成void类型,故就可以用(void(*)())来进行强制类型转换。(* (强制类型转换)0)();即(*(void (*)())0) ();
这是一个合法的C语言语句!
其意思是执行0地址空间的函数。
当然执行这样操作是非法的,就象引用空指针中的数据一样的概念!
如:int *p; *p=123; //语法正确,但是操作非法
解读这个语句:
1. 语句原型 (* ( void(*)() ) 0 )();
2. typedef void(*pfunc)() ; 定义一个函数指针类型pfunc
3. 转换一下原语句:(* (pfunc) 0 ) (); <=> (* ( void(*)() ) 0 )();
现在看起来清晰一点了,就是执行一个函数,但这个函数的地址是0
关于解读这类代码,请LZ查阅相关资料:C语言复杂类型的定义,自己好好体会
哈哈
这个东西确实 很牛
先这样理解:
void(*)() 是个 返回值为 void; 参数为空 (这样些也成 void(*)(void) )的函数指针
注意 是个 函数指针
然后 把0 强制转换为 这个类型的指针
然后 取 值 *(void(*)())
这个时候 就是 取得了 一个函数指针 类型为 返回值为 void; 参数为空
然后 后面加上 括号 就是 实施这个 函数指针的调用
大概系统编程时用到 其他 不用 不必深究