首页 > 技术文章 > 理解大端和小端

yejianyong 2017-07-03 19:33 原文

教你快速理解大端和小端
 
学习底层编程或逆向的童鞋,肯定对这两个名词并不陌生吧?!今天就给大家介绍一下这两个概念.
 
科普:
大端和小端,老外叫 Big-Endian  Little-Endian,其实指的都是同一个东东!在计算机界表示数据在存储器中的存放顺序不同的 CPU、操作系统对待数据的存储方式各有不同,但一般常见的操作系统都是小端,而通讯协议则是大端。
但并不是说系统是小端形式存储,文件就一定要采用小端的形式,不同的应用程序对于自身数据的存储方式也各有千秋(自家数据爱咋放咋放,有些还打乱了加密呢),比如:
       *Adobe PS -- 大端
       *BMP -- 小端
       *GIF -- 小端
       *JPEG -- 大端
       *MacPaint -- 大端
       *RTF -- 小端
那么小端和大端有什么区别呢?举个栗子大家就明了了:
比如 0x12345678 这个数:
       *大端法在内存中按字节依次存放为:12 34 56 78
       *小端法在内存中按字节依次存放为:78 56 34 12
 
解释:
大端:较高的有效字节存放在较低的存储器地址,较低的有效字节存放在较高的存储器地址。
比如整型变量 0x12345678  4 个字节,那么根据内存地址从小到大它们的存放方式如下:
  
数据
  
0x12
0x34
0x56
0x78
地址
0x10000000
0x10000001
0x10000002
0x10000003
小端:较高的有效字节存放在较高的的存储器地址,较低的有效字节存放在较低的存储器地址。所以整型变量0x12345678 根据内存地址从小到大它们的存放方式如下:
  
数据
  
0x78
0x56
0x34
0x12
地址
0x10000000
0x10000001
0x10000002
0x10000003
 
转换:
我们发现一个字节是可以存放两个十六进制的数字的(一个字节最大可以存放的数是 0xFF),那如果给你一个十进制数(比如 112233),如何快速地知道它在内存中是如何存放的呢?
l  大端法很容易,直接将它转换成十六进制,然后依次存放即可:0x0001B669
l  小端法则比较麻烦,步骤依次如下:
       *转换成十六进制数(0x0001B669
       *将八位数字的低四位和高四位互换(0xB6690001
       *在低四位和高四位中,分别进行两两互换(0x69B60100
如何检测你的机器是大端还是小端?
前边已经说过常见的个人电脑大多数都是使用小端但是我们都是我改变世界的程序猿不妨考虑小如何使用代码来分辨?
其实不难通过将int强转为插入单字节判断其实储存位置:
#include <stdio.h>
int main()
{
        int a = 0x2233;
        char *b =(char *)&a;
        if (*b ==0x22)
        {
                printf("大端!\n");
        }
        else
        {
               printf("小端!\n");
        }
        return 0;
}
<ignore_js_op>

推荐阅读