看到一篇关于C语言指针的文章,正好最近在看《C和指针》就去看了一下。《一道 C 语言指针访存题目的引申》,里面的第一个题目就把我难住了。
帮助123456789 #include <stdio.h> int main() { int a[5] = {1, 2, 3, 4, 5}; int *pa = (int)(&a) + 1; printf("%x\n", *pa); return 0; }
正确的输出结果应该是 2000000.自诩对C指针还比较了解,不就是保存内存地址的变量吗?有什么难的,后来才发现我错了。我想的是&a,是对数组名字去地址,那出来的结果不就是保存a的地址,那这样给他转换成int变量之后再加1,然后再赋值给一个int*指针是会出错的,结果我编译了一下,真的出错了:
error: invalid conversion from `int' to `int*'.
无法将int值赋值给int *指针变量,我修改了一下,在(int)(&a) + 1;前面加上了一个强制类型转换 (int *),编译之后输出结果:
我以为正确了,虽然他说正确答案是 2000000。后来我又仔细看了一下源代码,发现 int *pa = (int *)(int)(&a) + 1; 这句代码是先将 &a 转换成指针再在其基础加 1。而原来的意思是先给 &a 加 1再转换成指针,结果我再修改了一下代码:int *pa = (int *)((int)(&a) + 1); 这样结果竟然正确了,输出2000000:
这时候我完全朦了,这是怎么回事?后来我将 &a 和 a作为整数输出,想看看它到底是什么,结果发现
帮助12 printf("%d\n",&a); printf("%d\n",a);
他们的输出结果是一样的,也就是说a 和 &a 的值一样,我以为问题到这里就解决了,但是我还是不能理解为什么会输出 2000000.于是我到网上去搜索了一下资料,结果搜到了 对数组名取地址是什么? 。里面说
“
帮助123 1 int array[100]; 3 memset(array, 0, sizeof(array)); 4 memset(&array, 0, sizeof(array));
第3行和第4行有什么不同吗?其实从效果上来说是一样的,但是这里要注意 array 和 &array 的类型是不同的。array 相当于 &array[0],而 &array 是一个指向 int[100] 的指针,类型是 int(*)[100]。”
总算知道了a和&a的差别,但是我还是不明白为什么它会输出 2000000,这到底是为什么呢?
我试着修改了一下代码:
int a[] = {1,5};
这样之后的输出结果成了 : 5000000 原来和数组a的第二个元素有关系。哈哈,有点懂了,经过我N次的试验。
首先我们的代码是:
帮助123 int a[] = {1,2}; int *pa = (int *)((int)(&a) + 1); printf("%x\n", *(pa));
我这边编译的结果 a[0]也就是a代表的地址是 0x22ff70 a[1] 的地址是 0x22ff74 。我查看了一下内存(使用printf看的),内存里的数据是这样的:
帮助12 0x22ff70 0x22ff71 ... 0x22ff74 ... 0x22ff77 01 00 ... 02 ... 00
而且我们知道访问内存中的数据是高字节的地址是高位,低字节的地址是低位。也就是说如果我的地址是 0x22ff70,以这个地址访问一个整数,得到的整数就是73,72,71,70的数据排列 即00000001(以前学过一点汇编,加上自己试验了一下,就出来的 :),也就是1。如果访问0x22ff74 也就是 00000002 是2.如果访问 0x22ff71 也就是从74开始 02000000。这不就是我们的正确答案 2000000 。
问题解决!写得比较乱,勿怪。若有错误,欢迎指正!