*((int*)(&a+1) - 1)问题

分析如下代码:

int a[3] = {1,2,3};
printf("%d\n" , *((int*)(&a+1) - 1));
打印出来的结果如下:

    为什么?下面我们就来一一分析

    大家应该都知道,a的地址和&a[0]的地址一样,那么和&a的地址也一样。a和&a[0]是等同的,但是a和&a表达的意思却是不一样的。 a和&a[0]的类型都是 int *,即指向int的指针。而&a的类型是int *[3],即指向3个int元素的数组的指针。&a 这是一个指向数组a[3]的地址,即a[3]的第一个元素所在的地址。编译器将数组的地址默认为数组第一个元素的实际地址,也就是数组的首地址。

    &a 可以说是一个二级指针,那么(&a + 1)则移动的也是一个单元,但不是一个int单元,而是一个int[3]单位,就是整个数组都移动过去了。(&a + 1)也就是移动到a[3]的下一个单元位置了,在这里我们可以打印一下(&a + 1)的地址看一下,我们将会发现 (&a + 1)的地址和a的地址正好相差(sizeof(int) * 3)个字节的大小。

    那么(int *)(&a + 1)是什么意思呢,为什么在(&a + 1)前面还要加上(int *)呢,很简单(int *)这是一个强制类型转换,把这个二维二维的指针再强制转换成一维的。 

    (int *)(&a + 1) -1 也就是指的是将数组的地址再向前移动一个int单元。那么也就是指的是int[2]的位置。所以*((int *)(&a + 1) -1)打印出来的值就是a[2]的值 3 。


转载于:https://my.oschina.net/mjRao/blog/133839

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值