[转帖]C语言数组与指针_Android, Python及开发编程讨论区_Weblogic技术|Tuxedo技术|中间件技术|Oracle论坛|JAVA论坛|Linux/Unix技术|hadoop论坛_联动北方技术论坛  
网站首页 | 关于我们 | 服务中心 | 经验交流 | 公司荣誉 | 成功案例 | 合作伙伴 | 联系我们 |
联动北方-国内领先的云技术服务提供商
»  游客             当前位置:  论坛首页 »  自由讨论区 »  Android, Python及开发编程讨论区 »
总帖数
1
每页帖数
101/1页1
返回列表
0
发起投票  发起投票 发新帖子
查看: 3273 | 回复: 0   主题: [转帖]C语言数组与指针        下一篇 
linjun
注册用户
等级:中校
经验:2221
发帖:176
精华:2
注册:2013-1-6
状态:离线
发送短消息息给linjun 加好友    发送短消息息给linjun 发消息
发表于: IP:您无权察看 2013-1-9 14:35:04 | [全部帖] [楼主帖] 楼主

看到一篇关于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",&amp;a); printf("%d\n",a);

他们的输出结果是一样的,也就是说a 和 &a 的值一样,我以为问题到这里就解决了,但是我还是不能理解为什么会输出 2000000.于是我到网上去搜索了一下资料,结果搜到了 对数组名取地址是什么? 。里面说

帮助123 1 int array[100]; 3 memset(array,  0, sizeof(array)); 4 memset(&amp;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)(&amp;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 。
问题解决!写得比较乱,勿怪。若有错误,欢迎指正!
(全文完)




赞(0)    操作        顶端 
总帖数
1
每页帖数
101/1页1
返回列表
发新帖子
请输入验证码: 点击刷新验证码
您需要登录后才可以回帖 登录 | 注册
技术讨论