关于“网络字节序”与“主机字节序”

        近来,在进行的工作中涉及到许多网络编程的知识,特在此留下记录,是自己对相关知识的总结,也可作为后续对相关知识的回顾做个Mark。都说好记性不如烂笔头,觉得时常写写技术博客,不乏是一个组织知识架构的好方式。还是得坚持吧,争取从现在起,每周写一篇。我想对自己的职业生涯应该是很有帮助的。

1,理解各类字节序

        之前,搞混掉“大小端序”及“网络/主机字节序”的概念,认为它们是不同的概念,其实吧它们是一个事物的不同种说法罢了,它们中的详细术语解释及相互关系如下:

                大小端序:

                        1,小端序:地址低位存储值的低位,地址高位存储值的高位;(是最符合人的思维的字节序)
                        2,大端序:地址低位存储值的高位,地址高位存储值的低位。(是最直观的字节序)

                网络/主机字节序:

                        1,网络字节序:网络字节序是TCP/IP中规定好的一种数据表示格式,它与具体的CPU类型、操作系统等无关,从而可以保证数据在不同主机之间传输时

                              能够被正确解释。网络字节顺序采用big endian排序方式。

                        2,主机字节序:不同的 CPU 有不同的字节序类型,可能是大端序也可能是小端序,这些字节序是指整数在内存中保存的顺序。

        例子:如果我们将0x1234abcd写入到以0x0000开始的内存中,则结果为:

内存地址

0x0000

0x0001

0x0002

0x0003

Big-endian

0x12

0x34

0xab

0xcd

Little-endian

0xcd

0xab

0x34

0x12

        不同的CPU上运行不同的操作系统,字节序也是不同的,参见下表:

处理器

操作系统

字节排序

Alpha

全部

Little-endian

HP-PA

NT

Little-endian

HP-PA

UNIX

Big-endian

Intel-X86

全部

Little-endian

motorola680x()

全部

Big-endian

MIPS

NT

Little-endian

MIPS

UNIX

Big-endian

PowerPC

NT

Little-endian

PowerPC

非NT

Big-endian

RS/6000

UNIX

Big-endian

SPARC

UNIX

Big-endian

IXP1200 ARM核心

全部

Little-endian

2,网络编程中对字节序的使用

        在网络程序开发时或是跨平台开发时应该注意保证只用一种字节序不然两方的解释不一样就会产生bug。可以使用htonl、ntohl、htons、ntohs、BigLittleSwap16、BigLittleSwap32 这几个函数根据实际需要来实现字节序的转换。

        主机字节序转换为网络字节序时使用htonl,而在对网络字节序转换成主机字节序时使用ntohl。这样望文生义更容易理解!

具体实现如下(它们都是基于宏来实现的):

typedef unsigned short int uint16;
typedef unsigned long int uint32;

// 短整型大小端互换
#define BigLittleSwap16(A)        ((((uint16)(A) & 0xff00) >> 8) | \
                                                          (((uint16)(A) & 0x00ff) << 8))

// 长整型大小端互换
#define BigLittleSwap32(A)        ((((uint32)(A) & 0xff000000) >> 24) | \
                                                          (((uint32)(A) & 0x00ff0000) >> 8) | \
                                                          (((uint32)(A) & 0x0000ff00) << 8) | \
                                                          (((uint32)(A) & 0x000000ff) << 24))

// 本机大端返回1,小端返回0
int checkCPUendian()
{
        union{
                unsigned long int i;
                unsigned char s[4];
        }c = {0x12345678};

        return (0x12 == c.s[0]);
}

// 模拟htonl函数,本机字节序转网络字节序
#define HtoNl(A)        (checkCPUendian() ? (A) : BigLittleSwap32(A))

// 模拟ntohl函数,网络字节序转本机字节序
#define NtoHl(A)        HtoNl(A)

// 模拟htons函数,本机字节序转网络字节序
#define HtoNs(A)        (checkCPUendian() ? (A) : BigLittleSwap16(A))

// 模拟ntohs函数,网络字节序转本机字节序
#define NtoHs(A)        HtoNs(A)

总之理论还是得多结合实际才能对知识掌握地更深刻。

3,参考文献

        1,http://blog.csdn.net/tigerjb/article/details/6790294.

        2,http://wxxweb.blog.163.com/blog/static/135126900201031411125657/.

        3,  http://en.wikipedia.org/wiki/Endianness.

        4,  http://people.cs.umass.edu/~verts/cs32/endian.html.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值