(*(void(*)())0)() 解释

来源:互联网 发布:办公室网络共享 编辑:程序博客网 时间:2024/06/05 23:02

     首先具备知识:

       如何声明一个给定类型的标量,那么该类型的类型转换就很容易得到了:只需要把声明中的变量名 和声明末尾的分号去掉,再将剩余的部分用一个括号整个“ 封装”起来即可。例如:因为下面的声明:
                              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) ();