判断cpu大小端

一、大小端解析

        1、端模式出自Jonathan Swift书写的《格列佛游记》一书,这本书根据将鸡蛋敲开的方法不同将所有的人分为两类,从圆头开始将鸡蛋敲开的人被归为Big Endian,从尖头开始将鸡蛋敲开的人被归为Littile Endian。小人国的内战就源于吃鸡蛋时是究竟从大头(Big-Endian)敲开还是从小头(Little-Endian)敲开。

        2、  为什么会有大小端模式之分呢?这是因为在计算机系统中,我们是以字节为单位的,每个地址单元都对应着一个字节,一个字节为 8bit。但是在C语言中除了8bit的char之外,还有16bit的short型,32bit的long型(要看具体的编译器),另外,对于位数大于 8位的处理器,例如16位或者32位的处理器,由于寄存器宽度大于一个字节,那么必然存在着一个如何将多个字节安排的问题。因此就导致了大端存储模式和小 端存储模式。例如一个16bit的short型x,在内存中的地址为0x0010,x的值为0x1122,那么0x11为高字节,0x22为低字节。对于 大端模式,就将0x11放在低地址中,即0x0010中,0x22放在高地址中,即0x0011中。小端模式,刚好相反。字节。对于 大端模式,就将0x11放在低地址中,即0x0010中,0x22放在高地址中,即0x0011中。小端模式,刚好相反。

        2、在计算机业Big Endian和Little Endian也几乎引起一场战争。在计算机业界,Endian表示数据在存储器中的存放顺序。
        大端:高位存在低地址,低位存在高地址;这种方式是最直观的字节序

        小端:高位存在高地址,低位存在低地址;这种方式是最符合人的思维的字节序

       3、举个例子,从内存地址0x0000开始有以下数据
        0x0000    0x12
        0x0001    0x34
        0x0002    0xab
        0x0003    0xcd
    
        如果我们去读取一个地址为0x0000的四个字节变量:
                若字节序为big-endian,则读出结果为0x1234abcd;
                若字节序位little-endian,则读出结果为0xcdab3412.

        如果我们将0x1234abcd写入到以0x0000开始的内存中,则结果为:
                         big-endian      little-endian
        0x0000          0x12                  0xcd
        0x0001          0x23                  0xab
        0x0002          0xab                  0x34
        0x0003          0xcd                  0x12
        
        Intelx86系列以及ARM系列CPU都是little-endian的字节序.

二、 大小端判断

方法一:

<pre name="code" class="cpp">
void check_cpu()
{
    int i=1;  
    char *p=(char *)&i;  
    if(*p==1)    
           printf("Little_endian");   //Little_endian
    else
           printf("Big_endian");   //Big_endian
}
 

          大小端存储问题,如果小端方式中(i占四个字节的长度)则i所分配的内存最小地址那个字节中就存着1,其他字节是0.大端的话则1在i的最高地址字节处存放,char是一个字节,所以强制将char型量p指向i则p指向的一定是i的最低地址,那么就可以判断p中的值是不是1来确定是不是小端。

 方法二:

int check_cpu()
{
       union w
       {  
              int a;
              char b;
       } c;
       c.a = 1;\
       return(c.b ==1);
}

联合体union的存放顺序是所有成员都从低地址开始存放,可以利用这一点来写。
三、大小端转换
网络字节顺序NBO(Network Byte Order):按从高到低的顺序存储,在网络上使用统一的网络字节顺序,可以避免兼容性问题。
主机字节顺序HBO Host Byte Order): 不同的机器HBO不相同,与 CPU设计有关。
大端模式处理器的字节序到网络字节序不需要转换,此时ntohs(n)=n,ntohl =n;而小端模式处理器的字节序到网络字节必须要 进行转换(同理,有时候需要将大端字节顺序转换成小端字节顺序,也用这个函数,因为这个函数本来就是用来颠倒字节顺序的)。摘自:http://blog.csdn.net/yasaken/article/details/7243757
#if defined(BIG_ENDIAN) && !defined(LITTLE_ENDIAN)


   #define htons(A)  (A)
   #define htonl(A)  (A)
   #define ntohs(A)  (A)
   #define ntohl(A)  (A)


#elif defined(LITTLE_ENDIAN) && !defined(BIG_ENDIAN)


   #define htons(A)  ((((uint16_t)(A) & 0xff00) >> 8 ) | \\
                      (((uint16_t)(A) & 0x00ff) << 8 ))
   #define htonl(A)  ((((uint32_t)(A) & 0xff000000) >> 24)  | \\
                      (((uint32_t)(A) & 0x00ff0000) >> 8 )  | \\
                      (((uint32_t)(A) & 0x0000ff00) << 8 )  | \\
                      (((uint32_t)(A) & 0x000000ff) << 24))
   #define ntohs     htons
   #define ntohl     htohl


#else


   #error Either BIG_ENDIAN or LITTLE_ENDIAN must be #defined, but not both.


#endif

函数形式:
void convertToLittleEndian(unsigned int *data)
{
   *data = ((*data & 0xff000000) >> 24)
         | ((*data & 0x00ff0000) >>  8)
         | ((*data & 0x0000ff00) <<  8)
         | ((*data & 0x000000ff) << 24)
}



### 回答1: 大小端指的是数据在内存中存储的顺序,大端模式是高位字节存储在低地址,小端模式是高位字节存储在高地址。在Linux下,可以使用以下代码判断当前机器的大小端模式: ``` #include <stdio.h> int main() { unsigned int x = 0x12345678; char *c = (char*)&x; if (*c == 0x12) { printf("Big Endian\n"); } else { printf("Little Endian\n"); } return 0; } ``` ### 回答2: 在Linux中,判断系统是大端还是小端可以通过以下方法: 1. 使用C语言的方式判断:可以使用联合体的方式来判断大小端。可以创建一个包含一个char类型和一个int类型的联合体,然后将这个联合体的char和int成员的值分别设置为1,然后通过判断char成员的值来确定系统的字节序。若为1,则为小端;若为0,则为大端。 ```c #include <stdio.h> int main() { union { char c; int i; } endian; endian.i = 1; if (endian.c == 1) { printf("小端字节序\n"); } else { printf("大端字节序\n"); } return 0; } ``` 2. 使用命令行工具:在Linux终端中使用命令`lscpu`可以获取CPU的详细信息,其中有一项是"Byte Order",如果显示为"Little Endian"则表示是小端字节序,如果显示为"Big Endian"则表示是大端字节序。 综上所述,以上两种方法都可以在Linux中判断大小端。 ### 回答3: 在Linux中,可以通过以下方法判断当前系统的字节序是大端(Big Endian)还是小端(Little Endian)。 方法一:使用C语言编写一个小程序进行判断。 ```c #include<stdio.h> int main() { unsigned int num = 1; char *p = (char *)&num; if (*p == 1) { printf("该系统为小端字节序\n"); } else { printf("该系统为大端字节序\n"); } return 0; } ``` 该程序定义了一个无符号整型变量`num`并给其赋值为1,然后通过指针`p`来指向`num`的地址。由于整型在内存中是以字节的形式存储的,我们将首地址的字节与1进行比较,如果相等,则说明是小端字节序,否则为大端字节序。 方法二:使用命令行查看系统的字节序。 在Linux中,我们可以通过命令`lscpu`来查看系统的相关信息,其中包括字节序。在终端中输入如下命令: ``` lscpu | grep "Byte Order" ``` 如果输出结果为`Byte Order: Little Endian`,则说明系统为小端字节序。如果输出结果为`Byte Order: Big Endian`,则说明系统为大端字节序。 以上两种方法可以帮助我们在Linux中判断大小端
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值