小端与大端模式深度剖析

首先是对小端还有大端模式的剖析:

目前在各种体系的计算机中通常采用的字节存储机制主要有两种:Big-Endian和Little-Endian,下面先从字节序说起

一、什么是字节序
字节序,顾名思义字节的顺序,再多说两句就是大于一个字节类型的数据在内存中的存放顺序(一个字节的数据当然就无需谈顺序的问题了)。其实大部分人在实际的开 发中都很少会直接和字节序打交道。唯有在跨平台以及网络程序中字节序才是一个应该被考虑的问题。

a) Little-Endian就是低位字节排放在内存的低地址端,高位字节排放在内存的高地址端。
b) Big-Endian就是高位字节排放在内存的低地址端,低位字节排放在内存的高地址端。
c) 网络字节序:TCP/IP各层协议将字节序定义为Big-Endian,因此TCP/IP协议中使用的字节序通常称之为网络字节序。


ONE:什么是高低位地址?首先看下C程序映像中内存的空间布局情况:

------------------------------------------------------------------最高地址

                              

----------------------------------------------------------------------最高地址

可以看出 栈是向下增长的,堆是向上增长的!!

也就是说 先入栈的地址是比较高的,用一个简单的C程序可以验证一下

备注:入栈顺序和函数内的变量声明顺序可能不一样,这和编译器有一定的关系,所以不能通过一个函数内部的多个变量在栈中的顺序来判断栈的增长方向!!!我们可以利用函数调用时参数入栈的顺序来判断栈的增长顺序!

#include<iostream>
using namespace std;
//通过函数调用来判断栈的顺序

void fun2(int b)
{
	printf("%x\n",&b);
}


void fun1(int a)
{
	printf("%x\n",&a);
	fun2(a);
}

int	main()
{
	int a = 2;
	fun1(a);
}

程序结果为:


可以看出参数a的地址较高,所以栈是向下增长的

TWO:什么是高低字节

在十进制中我们都说靠左边的是高位,靠右边的是低位,在其他进制也是如此。就拿 0x12345678来说,从高位到低位的字节依次是0x12、0x34、0x56和0x78。

高/低地址端和高/低字节都弄清了。我们再来回顾 一下Big-Endian和Little-Endian的定义,并用图示说明两种字节序:


THREE:优缺点?

Big-Endian优点:靠首先提取高位字节,你总是可以由看看在偏移位置为0的字节来确定这个数字是 正数还是负数。你不必知道这个数值有多长,或者你也不必过一些字节来看这个数值是否含有符号位。这个数值是以它们被打印出来的顺序存放的,所以从二进制到十进制的函数特别有效。因而,对于不同要求的机器,在设计存取方式时就会不同。

Little-Endian优点:提取一个,两个,四个或者更长字节数据的汇编指令以与其他所有格式相同的方式进行:首先在偏移地址为0的地方提取最低位的字节,因为地址偏移和字节数是一对一的关系,多重精度的数学函数就相对地容易写了。

如果你增加数字的值,你可能在左边增加数字(高位非指数函数需要更多的数字)。 因此, 经常需要增加两位数字并移动存储器里所有Big-endian顺序的数字,把所有数向右移,这会增加计算机的工作量。不过,使用Little- Endian的存储器中不重要的字节可以存在它原来的位置,新的数可以存在它的右边的高位地址里。这就意味着计算机中的某些计算可以变得更加简单和快速。


写程序判断处理器是Big_endian还是Little_Endian

<span style="font-size:18px;">#include<iostream>
using namespace std;
//通过程序判断CPU是采用小端还是大端模式的存储
void checkCPU(void)
{
	union
	{
		int a;
		char b;
	}c;

	c.a = 1;//低位字节为1
	
	if(c.b == 1) //则说明地位字节放在低地址
		cout<<"Little endian"<<endl;
	else
		cout<<"Big endian"<<endl;
	
}
int main()
{
	checkCPU();
}</span>

或者用下面的一个简单的程序也行

<span style="font-size:18px;">#include<iostream>
using namespace std;
//通过程序判断CPU是采用小端还是大端模式的存储
void checkCPU()
{
	short int test = 0x1234;
	char *p = (char *)(&test);

	if(*p == 0x34)
	{

		cout<<"Little Endian"<<endl;
	}
	else
	{
		cout<<"Big Endian"<<endl;
	}
	
}
int main()
{
	checkCPU();
}</span>









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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值