### 主機字節序
主機字節序模式有兩種,大端數據模式和小端數據模式,在網絡編程中應注意這兩者的區別,以保證數據處理的正確性;例如網絡的數據是以大端數據模式進行交互,而我們的主
機大多數以小端模式處理,如果不轉換,數據會混亂 [參考](https://blog.csdn.net/dosthing/article/details/80641173);一般來說,兩個主機在網絡通
信需要經過如下轉換過程:主機字節序 —> 網絡字節序 -> 主機字節序
### 大端小端區別
- 大端模式:Big-Endian就是高位字節排放在內存的低地址端,低位字節排放在內存的高地址端
```bash
低地址 --------------------> 高地址
高位字節 地位字節
```
- 小端模式:Little-Endian就是低位字節排放在內存的低地址端,高位字節排放在內存的高地址端
```bash
低地址 --------------------> 高地址
低位字節 高位字節
```
#### 什么是高位字節和低位字節
例如在32位系統中,357轉換成二級制為:00000000 00000000 00000001 01100101,其中
```bash
00000001 | 01100101
高位字節 低位字節
```
### int和byte轉換
在go語言中,byte其實是uint8的別名,byte 和 uint8 之間可以直接進行互轉。目前來只能將0~255范圍的int轉成byte。因為超出這個范圍,go在轉換的時候,就會把多出來數據扔掉;如果需要將int32轉成
byte類型,我們只需要一個長度為4的[]byte數組就可以了
#### 大端模式下
```go
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]
}
```
#### 小端模式下
```go
// 在上面我們講過,小端剛好和大端相反的,所以在轉成小端模式的時候,只要將[]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 ]
}
```
### go轉換demo
```go
//整形轉換成字節
func IntToBytes(n int) []byte {
x := int32(n)
bytesBuffer := bytes.NewBuffer([]byte{})
binary.Write(bytesBuffer, binary.BigEndian, x)
return bytesBuffer.Bytes()
}
//字節轉換成整形
func BytesToInt(b []byte) int {
bytesBuffer := bytes.NewBuffer(b)
var x int32
binary.Read(bytesBuffer, binary.BigEndian, &x)
return int(x)
}
```
### 參考:
- [詳解大端模式和小端模式](https://www.cnblogs.com/little-white/p/3236548.html)
- [go實用小技能-int類型轉成byte類型原理解密](https://blog.csdn.net/m0_37191841/article/details/53908892)
- php
- 編譯安裝
- 基本概念
- 垃圾回收機制
- 生命周期
- zval底層實現
- c擴展開發
- gdb調試工具
- 自定義擴展簡單demo
- 鉤子函數
- 讀取php.ini配置
- 數組
- 函數
- 類
- yaf擴展底層源碼
- swoole擴展底層源碼
- memoryGlobal內存池
- swoole協程使用記錄
- 單點登錄sso原理
- compser使用
- session實現機制
- c & linux
- gcc
- 指針
- 結構體,聯合和位字段
- 宏定義井號說明
- printf家族函數和可變參數
- 共享函數
- 靜態庫和動態庫
- makefile自動化構建
- 信號一
- 信號二
- inotify監控文件事件
- socket編程
- 簡介
- UNIX DOMAIN
- Internet DOMAIN
- TCP/IP
- 文件IO多路復用
- 內存管理
- 進程組,會話和控制終端
- daemon守護進程
- 多進程
- 多線程
- 常用進制轉換
- go
- 入門知識
- 字節和整數裝換
- python
- redis
- 應用場景
- 消息隊列
- 熱點數據
- 掃碼登錄
- 訂閱發布
- 次數限制
- 搶購超賣
- 持久化機制
- mysql
- 工作流程
- MyISAM和InnoDB區別
- 用戶和權限管理
- 執行計劃
- sql優化
- 事務和鎖
- 慢查詢日志
- case...when...then...end用法
- sql
- 參考
- linux
- 內核參數優化
- 防火墻設置
- docker
- docker入門知識
- 算法
- 多維數組合
- DFA算法
- 紅包金額分配