地址的强制转换和大小端的两种求法

地址的强制转换和其转换的类型有很大的关系
下面来看《C语言深度解剖》上面的一道很有意思的题。

int main()
{
	int a[4]={1,2,3,4};
	int *ptr1=(int*)(&a+1);
	int *ptr2=(int*)((int)a+1);
	printf("%x,%x",ptr1[-1],*ptr2);
	return 0;
}

问:在X86系统下,其输出为多少?
1.&a+1和a+1分别代表什么意思?
我们知道a是数组首元素的地址,而&a代表的是取出整个数组的地址,虽然其值一样但是代表的意义是不同的。除了sizeof(数组名)和&(数组名)之外,所有的数组名都表示数组首元素的地址。
2.强制转换
上面我们知道了ptr1=&a+1其实就等于ptr1=&a+sizeof(int)*4;而ptr2=(int)a+1=a+(int)1;假设起始地址为0x00000000
假设起始地址为0x00000000。ptr2中a被强制转化为int型,(int)a+1就等于int(0x00000000)+1 = 0+1。一个地址搭配一个字节。
3.机器的大小端问题。
我们由上图知道了数组a在内存中的排列和ptr1、ptr2指向哪里。但是要知道这个代码的结果还得看你的机器到底是大端还是小端了。这可真是个鸡儿问题,很多人在这里都跳到坑里面了。
小端无非就是低地址存低字节,高地址存高字节;大端是低地址存高字节,高地址存低字节内存中地址是线性排列的,并且是由低到高排列的。
4.求大小端的两种方法
(1)利用指针的特性

int sys_check()
{
	int i=1;
	char *c=&i;
	return (*c);
}
int main()
{
	int ret=sys_check();
	if(ret==1)
	{
		printf("小端";
		}
	else
	{
		printf("大端";
		}
}

指针的类型是多少字节的,则在其解引用时就会解引用多少字节的数据。(这句话不太严谨)
(2)利用共用体(联合体)

int sys_check()
{
	union check
	{
		int i;
		char ch;
	}c;
	c.i=1;
	return (c.ch==1)	//如果ch里面是1的话,那么c.ch==1成立,也就是return 1
}
int main()
{
	int ret=sys_check();
	if(ret==1)
	{
		printf("小端";
		}
	else
	{
		printf("大端";
		}
}

利用共用体变量公用同一块内存的特性。
好了,终于所有的分析工作都做完了。那么当系统是小端的时候,*ptr2=0x02000000,ptr1[-1]=0x00000010;大端时:*ptr2=0x00000100,ptr1[-1]=0x00000010。

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值