谷动谷力

 找回密码
 立即注册
查看: 648|回复: 0
打印 上一主题 下一主题
收起左侧

大端、小端基础知识 什么是大小端?

[复制链接]
跳转到指定楼层
楼主
发表于 2024-2-27 16:17:35 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式
大端、小端基础知识 什么是大小端?


一、什么是大小端?
对于一个由2个字节组成的16位整数,在内存中存储这两个字节有两种方法:一种是将低序字节存储在起始地址,这称为小端(little-endian)字节序;另一种方法是将高序字节存储在起始地址,这称为大端(big-endian)字节序。



假如现有一32位int型数0x12345678,那么其MSB(Most Significant Byte,最高有效字节)为0x12,其LSB (Least Significant Byte,最低有效字节)为0x78,在CPU内存中有两种存放方式:(假设从地址0x4000开始存放)



总结:

大端是高字节存放到内存的低地址

小端是高字节存放到内存的高地址

二、如何确定大小端
有些CPU公司用大端(譬如C51单片机);有些CPU用小端(譬如ARM)。(大部分是用小端模式,大端模式的不算多)。于是乎我们写代码时,当不知道当前环境是用大端模式还是小端模式时就需要用代码来检测当前系统的大小端。下面给出用程序判断大小端的两种方法:

  1. #include <stdio.h>

  2. // 共用体中很重要的一点:a和b都是从u1的低地址开始存放的。
  3. // 假设u1所在的4字节地址分别是:0、1、2、3的话,那么a自然就是0、1、2、3;
  4. // b所在的地址是0而不是3.

  5. union myunion
  6. {
  7.         int a;
  8.         char b;
  9. };

  10. // 如果是小端模式则返回1,小端模式则返回0
  11. int is_little_endian(void)
  12. {
  13.         union myunion u1;
  14.         u1.a = 0x12345678;                                // 地址0的那个字节内是0x78(小端)或者0x12(大端)
  15.     if(0x78 == u1.b)
  16.         return 1;
  17.     else if(0x12 == u1.b)
  18.             return 0;
  19. }

  20. int is_little_endian2(void)
  21. {
  22.         int a = 0x12345678;
  23.         char b = *((char *)(&a));                // 指针方式其实就是共用体的本质
  24.         if(0x78 == b)
  25.         return 1;
  26.     else if(0x12 == b)
  27.             return 0;
  28. }


  29. int main(void)
  30. {
  31.         int i = is_little_endian2();
  32.         //int i = is_little_endian();
  33.         if (i == 1)
  34.         {
  35.                 printf("小端模式\n");
  36.         }
  37.         else
  38.         {
  39.                 printf("大端模式\n");
  40.         }
  41.         return 0;
  42. }
复制代码


下面给出三种确定大小端错误的方案
  1. // 强制类型转换
  2. int a;
  3. char b;
  4. a = 1;
  5. b = (char)a;
  6. printf("b = %d.\n", b);
  7.         // 移位
  8. int a, b;
  9. a = 1;
  10. b = a >> 1;
  11. printf("b = %d.\n", b);
  12.         // 位与
  13. int a = 1;
  14. int b = a & 0xff;                // 也可以写成:char b = a & 0x01;
  15. printf("b = %d.\n", b);
复制代码


位与、移位、强制类型转换等运算是编译器提供的运算,这个运算是高于内存层次的(或者说这些运算在二进制层次具有可移植性,也就是说&的时候一定是高字节&高字节,低字节&低字节,和二进制存储无关)。

+10
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|Archiver|手机版|深圳市光明谷科技有限公司|光明谷商城|Sunshine Silicon Corpporation ( 粤ICP备14060730号|Sitemap

GMT+8, 2024-5-5 16:23 , Processed in 0.131788 second(s), 43 queries .

Powered by Discuz! X3.2 Licensed

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表