网络字节序

一.字节序

       字节序是由于不同的主处理器和操作系统,对大于一个字节的变量在内存中的存放顺序不同而产生的。

       字节序通常有大端字节序列和小端字节序两种分类方法。

        由于主机的千差万别,主机的字节序不能做到统一,但是网络上传输的数值,它们有统一的规定。网络字节序:是指多字节变量在网络传输时的表示方法,网络字节序采用高端字节序的表示方法

二.大端字节序和小端字节序

    字节序是由于CPU和OS对多字节变量的内存存储顺序不同而产生的。

1.brief introduce:

    一个16位的整数,它由两个字节构成,在有的系统上会将高字节放在内存的低地址上,而有的系统上则将高字节放在内存的高地址上,所以存在字节序的问题。

2.字节序的表示方法有两种:

1>小端字节序(Little Endian):在表示变量的内存起始地址存放底字节,高字节顺序存放

2>大端字节序(Big Endian):在表示变量的内存起始地址存放高字节,底字节顺序存放。

3实例:

一个变量的值为0xabcd,在大端字节序和小端字节序的系统中两者的存放顺序是不同的.

1>在小端字节系统中的存放顺序如图:

网络字节序_byte

0xab在地址15~8的地址上,而0xcd在地址7~0的位置上。

2>在大端字节序系统中的存放顺序如图:

网络字节序_网络_02

4.程序实例:

可以编写程序来检测下变量在内存中的表示形式,以确定系统中的字节序为大端字节序还是小端字节序。

(1)字节序结构:

程序先建立一个公用体类型address,用于测试字节序,成员value是int 类型变量,可以通过成员byte来访问value变量的高字节和低字节。

typedef union{

   unsigned  int value;

   unsigned  char byte[2];

}addr;

(2)变量声明,声明一个to类型的变量typeordr,给变量value赋值为0xabcd,由于在类型addr中,value和byte成员共享同一块内存,所以可以通过byte的不同成员来访问value的高字节和低字节。

(3)程序:

int main(int argc,char *argv)

{

      addr  typeorder;//一个to类型的变量

      typeorder.value= 0xabcd;

      /*小端字节序检查*/

       if(typeorder.byte[0] == 0xcd &&typeorder.byte[1]==0xab){

        printf(“Low endian bye order\n”);

        printf(“byte[0]:0x%x,byte[1]:0x%x\n”,typeorder.byte[0],typeorder.byte[1]);          

         }

           /*大端字节序判断*/

          if(typeorder.byte[0] == 0xab &&typeorder.byte[1]==0xcd){

          printf(“High endian byte order\n”);

            printf(“byte[0]:0x%x,byte[1]:0x%x\n”,typeorder.byte[0],typeorder.byte[1]);

           }

          return 0;

}

5.进行网络字节序转换的函数有htons,htonl,ntohs(),ntohl()等,其中s表示short数据类型,l是long数据类型的意思,h是host既主机的意思,n是network即网络的意思。

1>htons(host to net short):表示对short类型的变量,从主机字节序转换为网络字节序。

2>ntohs(net to host short):表示对short类型的变量,从网络字节序转换为主机字节序。

3>htonl(host to net long):表示对long类型的变量,从主机字节序转换为网络字节序

4>ntohl(net to host long):表示对long类型的变量,从网络字节序转换为主机字节序。

6.由于在网络中网络字节序采用高端字节序的表示方法。因此字节序转换函数在不同平台上的实现是不同的,在小端主机字节序的平台要进行转换,而在大端主机字节序的平台上是不需要进行转换的。

例如:通过下面的实现方式可以兼容不同的平台:

#if  ISLE

/*小端字节序平台调用此部分代码*/

long  htonl(longvalue)

{

Return ((value << 24) |  ((value <<8)&0x00ff0000) |((value>>8)&0x0000ff00)|(value>>24));

}

#else if  ISBE

/*大端字节序平台调用此部分代码*/

long  htonl(longvalue)

{

      /*由于字节序一致,不需要进行转换*/

      return  value;

}

#endif