在例2.11中,既然&a[i]表示数组a的各个元素的地址,这些地址存储相应数组元素的值,就应该能直接将这些值输出。
【例2.13】使用“*”操作符输出数组内容。
#include <stdio.h>void main ( ){ int a[3]={1,2,3}; int i=0; //输出数组地址 for(i=0;i<3;i++) printf("0x%p ",&a[i]); printf("/n"); //输出数组内容 for(i=0;i<3;i++) printf("%d ",*&a[i]); printf("/n");}
地址用十六进制输出,地址里的内容采用在地址前加“*”号的方法输出。输出结果如下。
0x0012FF74 0x0012FF78 0x0012FF7C1 2 3
上面演示了对数组的地址使用“*”操作的方法。如果用变量存储一个有效的地址,例如整型变量addr存储整型变量a的地址,能用“*addr”输出变量a的值吗?下面就通过程序来看看是否可行。
【例2.14】分析程序中存在的错误。
#include <stdio.h>int main(){ int a=25; int addr; addr=&a; printf("分配给变量a的地址0x%d/n",&a); printf("addr的地址值=0x%p/n",addr); //输出addr的值 printf("addr地址里的内容=%d/n",*addr); //输出addr的内容 return 0;}
编译给出一个错误信息和一个警告信息。
warning C4047: '=' : 'int ' differs in levels of indirection from 'int *'error C2100: illegal indirection
因为addr是整型变量,&a是地址值,所以造成语句“=”两边的数据类型不匹配。可以对地址值使用int进行强制转换以解决不匹配问题,即使用语句
addr=(int)&a;
将地址转换成整型数值,完成“=”两边的匹配,这样就解决了编译时的警告信息。
直接对&a使用“*”,输出正确。但对addr出错,显然也是表达方式不匹配。使用强制转换,改用
printf("addr地址里的内容=%d/n",(int*)addr);
编译正确,输出结果如下。
分配给变量a的地址0x1245052addr的地址值=0x0012FF7Caddr地址里的内容=1245052
虽然输出结果不对,但有了解决办法。“1245052”就是addr的十进制地址,所以对这个地址使用“*”操作符即可。
因为语句“addr=(int)&a;”将地址强制转换成int类型值的地址,所以不能直接使用“*addr”,需要使用“(int*)addr”再将其转换成存储的地址值。程序的输出结果也证实了它就是变量a的地址,也就是存储在addr变量里的地址。这时再对它使用“*(int*)addr”,就能输出存储在addr地址里的值了。
//改正后的正确程序#include <stdio.h>int main(){ int a=25; int addr; addr=(int)&a; printf("分配给变量a的地址0x%d/n",&a); printf("addr的地址值=0x%p/n",addr); //输出addr的值 printf("addr地址里的内容=%d/n",*&a); //输出addr的值 printf("addr地址里的内容=%d/n",*(int*)addr); //输出addr的内容 return 0;}
输出结果如下。
分配给变量a的地址0x1245052addr的地址值=0x0012FF7Caddr地址里的内容=25addr地址里的内容=25
显然,把变量a的地址直接赋给变量addr,效果是一样的。
【例2.15】直接将地址赋给变量的例子。
#include <stdio.h>int main(){ int a=25; int addr; addr=(int)0x0012FF7C; printf("分配给变量a的地址0x%d/n",&a); printf("addr的地址值=0x%p/n",addr); //输出addr的值 printf("addr地址里的内容=%d/n",*&a); //输出addr的值 printf("addr地址里的内容=%d/n",*(int*)addr); //输出addr的内容 return 0;}
运行结果相同,不再给出。