国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Golang > 正文

golang網絡socket粘包問題的解決方法

2020-04-01 19:10:50
字體:
來源:轉載
供稿:網友

本文實例講述了golang網絡socket粘包問題的解決方法。分享給大家供大家參考,具體如下:

看到很多人問這個問題, 今天就寫了個例子, 希望能幫助大家

首先說一下什么是粘包:百度上比較通俗的說法是指TCP協議中,發送方發送的若干包數據到接收方接收時粘成一包,從接收緩沖區看,后一包數據的頭緊接著前一包數據的尾

解決方案如下:

服務端:

復制代碼 代碼如下:
package main
import (
    "bytes"
    "encoding/binary"
    "fmt"
    "io"
    "net"
)
func main() {
    // 監聽端口
    ln, err := net.Listen("tcp", ":6000")
    if err != nil {
        fmt.Printf("Listen Error: %s/n", err)
        return
    }
    // 監聽循環
    for {
        // 接受客戶端鏈接
        conn, err := ln.Accept()
        if err != nil {
            fmt.Printf("Accept Error: %s/n", err)
            continue
        }
        // 處理客戶端鏈接
        go handleConnection(conn)
    }
}
func handleConnection(conn net.Conn) {
    // 關閉鏈接
    defer conn.Close()
    // 客戶端
    fmt.Printf("Client: %s/n", conn.RemoteAddr())
    // 消息緩沖
    msgbuf := bytes.NewBuffer(make([]byte, 0, 10240))
    // 數據緩沖
    databuf := make([]byte, 4096)
    // 消息長度
    length := 0
    // 消息長度uint32
    ulength := uint32(0)
    // 數據循環
    for {
        // 讀取數據
        n, err := conn.Read(databuf)
        if err == io.EOF {
            fmt.Printf("Client exit: %s/n", conn.RemoteAddr())
        }
        if err != nil {
            fmt.Printf("Read error: %s/n", err)
            return
        }
        fmt.Println(databuf[:n])
        // 數據添加到消息緩沖
        n, err = msgbuf.Write(databuf[:n])
        if err != nil {
            fmt.Printf("Buffer write error: %s/n", err)
            return
        }
        // 消息分割循環
        for {
            // 消息頭
            if length == 0 && msgbuf.Len() >= 4 {
                binary.Read(msgbuf, binary.LittleEndian, &ulength)
                length = int(ulength)
                // 檢查超長消息
                if length > 10240 {
                    fmt.Printf("Message too length: %d/n", length)
                    return
                }
            }
            // 消息體
            if length > 0 && msgbuf.Len() >= length {
                fmt.Printf("Client messge: %s/n", string(msgbuf.Next(length)))
                length = 0
            } else {
                break
            }
        }
    }
}

 

客戶端:

復制代碼 代碼如下:
package main
import (
    "bytes"
    "encoding/binary"
    "fmt"
    "net"
    "time"
)
func main() {
    // 鏈接服務器
    conn, err := net.Dial("tcp", "127.0.0.1:6000")
    if err != nil {
        fmt.Printf("Dial error: %s/n", err)
        return
    }
    // 客戶端信息
    fmt.Printf("Client: %s/n", conn.LocalAddr())
    // 消息緩沖
    msgbuf := bytes.NewBuffer(make([]byte, 0, 1024))
    // 消息內容
    message := []byte("我是utf-8的消息")
    // 消息長度
    messageLen := uint32(len(message))
    // 消息總長度
    mlen := 4 + len(message)
    // 寫入5條消息
    for i := 0; i < 10; i++ {
        binary.Write(msgbuf, binary.LittleEndian, messageLen)
        msgbuf.Write(message)
    }
    // 單包發送一條消息
    conn.Write(msgbuf.Next(mlen))
    time.Sleep(time.Second)
    // 單包發送三條消息
    conn.Write(msgbuf.Next(mlen * 3))
    time.Sleep(time.Second)
    // 發送不完整的消息頭
    conn.Write(msgbuf.Next(2))
    time.Sleep(time.Second)
    // 發送消息剩下部分
    conn.Write(msgbuf.Next(mlen - 2))
    time.Sleep(time.Second)
    // 發送不完整的消息體
    conn.Write(msgbuf.Next(mlen - 6))
    time.Sleep(time.Second)
    // 發送消息剩下部分
    conn.Write(msgbuf.Next(6))
    time.Sleep(time.Second)
    // 多段發送
    conn.Write(msgbuf.Next(mlen + 2))
    time.Sleep(time.Second)
    conn.Write(msgbuf.Next(-2 + mlen - 8))
    time.Sleep(time.Second)
    conn.Write(msgbuf.Next(8 + 1))
    time.Sleep(time.Second)
    conn.Write(msgbuf.Next(-1 + mlen + mlen))
    time.Sleep(time.Second)
    // 關閉鏈接
    conn.Close()
}

 

希望本文所述對大家Go語言程序設計有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 客服| 青岛市| 天台县| 丹寨县| 安新县| 八宿县| 平谷区| 河南省| 泽州县| 白水县| 娄烦县| 会昌县| 兴山县| 梁平县| 祥云县| 古蔺县| 佛教| 永善县| 祁阳县| 肥西县| 鄂伦春自治旗| 涞水县| 陇西县| 昌邑市| 商南县| 邛崃市| 盘山县| 东阳市| 贵溪市| 涟源市| 枣阳市| 长岛县| 河南省| 榆社县| 天津市| 柘城县| 商洛市| 绥芬河市| 永和县| 绥中县| 内江市|