理解字节序
<p>1.</p> <p>计算机硬件有两种储存数据的方式:大端字节序(big endian)和小端字节序(little endian)。</p> <p>举例来说,数值 0x2211 使用两个字节储存:高位字节是 0x22 ,低位字节是 0x11 。</p> <ul> <li><strong>大端字节序</strong> :高位字节在前,低位字节在后,这是人类读写数值的方法。</li> <li><strong>小端字节序</strong> :低位字节在前,高位字节在后,即以 0x1122 形式储存。</li> </ul> <p style="text-align: center;"><img src="https://simg.open-open.com/show/c26fa89ad02b0286cc3d78729fb017ac.jpg"></p> <p>同理, 0x1234567 的大端字节序和小端字节序的写法如下图。</p> <p style="text-align: center;"><img src="https://simg.open-open.com/show/1134e1c02799d5704e55c10f531f2a53.gif"></p> <p>2.</p> <p>我一直不理解,为什么要有字节序,每次读写都要区分,多麻烦!统一使用大端字节序,不是更方便吗?</p> <p>上周,我读到了一篇 文章 ,解答了所有的疑问。而且,我发现原来的理解是错的,字节序其实很简单。</p> <p>3.</p> <p>首先,为什么会有小端字节序?</p> <p>答案是,计算机电路先处理低位字节,效率比较高,因为计算都是从低位开始的。所以,计算机的内部处理都是小端字节序。</p> <p>但是,人类还是习惯读写大端字节序。所以,除了计算机的内部处理,其他的场合几乎都是大端字节序,比如网络传输和文件储存。</p> <p>4.</p> <p>计算机处理字节序的时候,不知道什么是高位字节,什么是低位字节。它只知道按顺序读取字节,先读第一个字节,再读第二个字节。</p> <p>如果是大端字节序,先读到的就是高位字节,后读到的就是低位字节。小端字节序正好相反。</p> <p>理解这一点,才能理解计算机如何处理字节序。</p> <p>5.</p> <p>字节序的处理,就是一句话:</p> <p><strong>"只有读取的时候,才必须区分字节序,其他情况都不用考虑。"</strong></p> <p>处理器读取外部数据的时候,必须知道数据的字节序,将其转成正确的值。然后,就正常使用这个值,完全不用再考虑字节序。</p> <p>即使是向外部设备写入数据,也不用考虑字节序,正常写入一个值即可。外部设备会自己处理字节序的问题。</p> <p>6.</p> <p>举例来说,处理器读入一个16位整数。如果是大端字节序,就按下面的方式转成值。</p> <pre> x = buf[offset] * 256 + buf[offset+1];</pre> <p>上面代码中, buf 是整个数据块在内存中的起始地址, offset 是当前正在读取的位置。第一个字节乘以256,再加上第二个字节,就是大端字节序的值,这个式子可以用逻辑运算符改写。</p> <pre> x = buf[offset]<<8 | buf[offset+1];</pre> <p>上面代码中,第一个字节左移8位(即后面添8个 0 ),然后再与第二个字节进行或运算。</p> <p>如果是小端字节序,用下面的公式转成值。</p> <pre> x = buf[offset+1] * 256 + buf[offset];</pre> <p>32位整数的求值公式也是一样的。</p> <pre> /* 大端字节序 */ i = (data[3]<<0) | (data[2]<<8) | (data[1]<<16) | (data[0]<<24); /* 小端字节序 */ i = (data[0]<<0) | (data[1]<<8) | (data[2]<<16) | (data[3]<<24);</pre> <p>(完)</p> <p> </p> <p>来自:http://www.ruanyifeng.com/blog/2016/11/byte-order.html</p> <p> </p>
本文由用户 gw8428 自行上传分享,仅供网友学习交流。所有权归原作者,若您的权利被侵害,请联系管理员。
转载本站原创文章,请注明出处,并保留原始链接、图片水印。
本站是一个以用户分享为主的开源技术平台,欢迎各类分享!