#include <stdio.h>
void main()
{
int *p;
int a=2;
unsigned long b=1245048;
p=&a;
printf("%d/n",*p);
printf("%p/n",&a);
printf("%d/n",&a);
printf("%d/n",(void*)b);
//printf("%d/n",*(void*)b); // 被注释的一行,运行此行会提示错误。
printf("%d/n",*(int*)b);
printf("%d/n",*(void**)b);
}
P 为指向整型变量的指针。将 a 的地址赋予 p ,打印出 *p (解引用,即取该地址的值)为 2 。
再打印出 &a ,即 a 地址,为 0012FF78 (参数 p 表示用指针的格式,即内存地址,打印出来)。
再打印出 &a 的地址的十进制数字为 1245048( 参数 d 表示为十进制 ) 。
定义一个整型变量 b ,赋值为 1245048 ( 即 unsigned long b=1245048;) 。
如此, printf("%d/n",(void*)b); 表示:将 b 强制转换为一个指针,并打印出来。结果是 1245048 。
为什么如此呢?因为此时 b 虽然转换为一个指针,但 printf 的时候却没加上 * 号,故此没进行解引用(也即是取该地址的值),因而打印出来的仍然是该指针的值。(注意区别指针的值,和该指针指向的地址的值)
那么,加上 * 号进行解引用再打印出来,可否?实验结果不行,出现语法错误。概因 b 只是强制转换为指针,并没有指明是什么样类型(其实是 void 型)的指针。
那么,强制转换的时候便声明该指针类型,可以吗?可以看看 printf("%d/n",*(int*)b); ,将 b 强制转换为 int* 类型的指针,解引用成功,输出 2 。
最后来到我们的问题, *(void**)b 究竟是什么?相信看到这里你大概知道了。
(void**) 代表的是指向指针的指针,如此,先 (void**)b ,即,将 b 强制转换为指向指针的指针,然后再给它加上一个 * 解引用,如此,便取得了该指针指向的地址的值, 2 。
你可能会说,这里不也是声明为 void 型的指针了吗?这里行为什么上边不行呢?不,这里其实是不一样的。上边是void 型指针,本质即指针,而这里是 void 型的指针的指针,本质是指针的指针。 void 型指针,不知道指向的地址内容要怎么去引用它,而 void 型的指针的指针,却知道其指向的地址内容是一个地址。
当然,我们也可以 printf("%d/n",*(int**)b); 输出结果是一样的。因为指向指针的指针本来就是指针(恩,有点拗口),只要类型正确,决定该指针输出什么的是其指向哪里的值,而不是指针本身。
我们可以做个试验, printf("%d/n",*(int****)b); ,看起来很晕吧,将 b 转换为指向指针的指针的指针的指针,结果呢?一样都是 2 。