1.主機字節(jié)序
主機字節(jié)序模式有兩種释牺,大端數據模式和小端數據模式髓削。在網絡編程中應注意這兩者的區(qū)別才写,以保證數據處理的正確性葡兑。例如,網絡的數據是以大端數據模式進行交互赞草,而我們的主機大多數以小端模式處理讹堤,如果不轉換,數據會混亂 厨疙。
參考:一般來說洲守,兩個主機在網絡通信需要經過如下轉換過程:主機字節(jié)序 -> 網絡字節(jié)序 -> 主機字節(jié)序
2.大端小端區(qū)別
大端模式:Big-Endian就是高位字節(jié)排放在內存的低地址端,低位字節(jié)排放在內存的高地址端
低地址 --------------------> 高地址
高位字節(jié) 地位字節(jié)
小端模式:Little-Endian就是低位字節(jié)排放在內存的低地址端,高位字節(jié)排放在內存的高地址端
低地址 --------------------> 高地址
低位字節(jié) 高位字節(jié)
2.1 什么是高位字節(jié)和低位字節(jié)梗醇?
例如在32位系統(tǒng)中知允,357轉換成二級制為:
00000000 00000000 00000001 01100101,其中
00000001 | 01100101
高位字節(jié) 低位字節(jié)
3.int與byte類型轉換
在GO語言中叙谨,byte其實是uint8的別名温鸽,byte 和 uint8 之間可以直接進行互轉。目前手负,只能將0 ~ 255范圍的int轉成byte涤垫。因為超出這個范圍,GO在轉換的時候竟终,就會把多出來數據扔掉蝠猬。如果需要將int32轉成byte類型,我們只需要一個長度為4的[]byte數組就可以统捶。
3.1 大端模式下
func f2() {
var v2 uint32
var b2 [4]byte
v2 = 257
// 將 257轉成二進制就是
// | 00000000 | 00000000 | 00000001 | 00000001 |
// | b2[0] | b2[1] | b2[2] | b2[3] | // 這里表示b2數組每個下標里面存放的值
// 這里直接使用將uint32強轉成uint8
// | 00000000 0000000 00000001 | 00000001 直接轉成uint8后等于 1
// |---這部分go在強轉的時候扔掉---|
b2[3] = uint8(v2)
// | 00000000 | 00000000 | 00000001 | 00000001 | 右移8位 轉成uint8后等于 1
// 下面是右移后的數據
// | | 00000000 | 00000000 | 00000001 |
b2[2] = uint8(v2 >> 8)
// | 00000000 | 00000000 | 00000001 | 00000001 | 右移16位 轉成uint8后等于 0
// 下面是右移后的數據
// | | | 00000000 | 00000000 |
b2[1] = uint8(v2 >> 16)
// | 00000000 | 00000000 | 00000001 | 00000001 | 右移24位 轉成uint8后等于 0
// 下面是右移后的數據
// | | | | 00000000 |
b2[0] = uint8(v2 >> 24)
fmt.Printf("%+v\n", b2)
// 所以最終將uint32轉成[]byte數組輸出為
// [0 0 1 1]
}
3.2 小端模式下
// 在上面我們講過榆芦,小端剛好和大端相反的,所以在轉成小端模式的時候喘鸟,只要將[]byte數組的下標首尾對換一下位置就可以了
func f3() {
var v3 uint32
var b3 [4]byte
v3 = 257
// 將 256轉成二進制就是
// | 00000000 | 00000000 | 00000001 | 00000001 |
// | b3[0] | b3[1] | b3[2] | [3] | // 這里表示b3數組每個下標里面存放的值
// 這里直接使用將uint32l強轉成uint8
// | 00000000 0000000 00000001 | 00000001 直接轉成uint8后等于 1
// |---這部分go在強轉的時候扔掉---|
b3[0] = uint8(v3)
// | 00000000 | 00000000 | 00000001 | 00000001 | 右移8位 轉成uint8后等于 1
// 下面是右移后的數據
// | | 00000000 | 00000000 | 00000001 |
b3[1] = uint8(v3 >> 8)
// | 00000000 | 00000000 | 00000001 | 00000001 | 右移16位 轉成uint8后等于 0
// 下面是右移后的數據
// | | | 00000000 | 00000000 |
b3[2] = uint8(v3 >> 16)
// | 00000000 | 00000000 | 00000001 | 00000001 | 右移24位 轉成uint8后等于 0
// 下面是右移后的數據
// | | | | 00000000 |
b3[3] = uint8(v3 >> 24)
fmt.Printf("%+v\n", b3)
// 所以最終將uint32轉成[]byte數組輸出為
// [1 1 0 0 ]
}
3.3 代碼示例
package main
import (
"bytes"
"encoding/binary"
"fmt"
)
//整形轉換成字節(jié)
func IntToBytes(n int) []byte {
x := int32(n)
bytesBuffer := bytes.NewBuffer([]byte{})
_ = binary.Write(bytesBuffer, binary.BigEndian, x)
return bytesBuffer.Bytes()
}
//字節(jié)轉換成整形
func BytesToInt(b []byte) int {
bytesBuffer := bytes.NewBuffer(b)
var x int32
_ = binary.Read(bytesBuffer, binary.BigEndian, &x)
return int(x)
}
func main() {
var a int
a = 20
b := []byte {0, 0, 0, 'A'}
fmt.Println(IntToBytes(a))
fmt.Println(BytesToInt(b))
}
輸出結果:
[0 0 0 20]
65
原貼:
https://studygolang.com/articles/16154
個人主頁: