首頁 >後端開發 >Golang >停用 keepalive 時 io.Copy 逾時

停用 keepalive 時 io.Copy 逾時

PHPz
PHPz轉載
2024-02-13 21:36:081240瀏覽

禁用 keepalive 时 io.Copy 超时

php小編魚仔今天要介紹一個關於"禁用 keepalive 時 io.Copy 超時"的問題。在使用Go語言中的io.Copy函數進行資料傳輸時,如果停用了keepalive,可能會導致傳輸逾時的問題。這個問題可能會給開發者帶來一些困擾,因此我們需要了解一些解決方法來避免這個問題的發生。下面就讓我們一起來看看如何解決這個問題吧!

問題內容

當我停用 linux keepalive 時

sudo sysctl -w net.ipv4.tcp_keepalive_probes=0

並執行以下程式碼

package main

import (
    "fmt"
    "io"
    "net"
    "os"
    "sync"
    "time"
)

func main() {
    go func() {
        listen, err := net.Listen("tcp", "127.0.0.1:9390")
        if err != nil {
            fmt.Printf("net listen fail, reason: [%s]\n", err.Error())
            os.Exit(1)
        }
        defer listen.Close()

        for {
            conn, err := listen.Accept()
            if err != nil {
                fmt.Printf("net accept fail, reason: [%s]\n", err.Error())
                continue
            }

            _, err = io.Copy(conn, conn)
            if err != nil {
                fmt.Printf("net Copy fail, reason: [%s]\n", err.Error())
                continue
            }
            conn.Close()
        }
    }()

    var wg sync.WaitGroup
    wg.Add(1)
    go func() {
        defer wg.Done()
        for {
            time.Sleep(1 * time.Second)
            tcpAddr, _ := net.ResolveTCPAddr("tcp", "127.0.0.1:9390")
            fmt.Println("tcp client resolve success: ", tcpAddr.String())
            tcpConn, err := net.DialTCP("tcp", nil, tcpAddr)
            if err != nil {
                fmt.Println("tcp connect fail: ", err.Error())
                return
            }

            time.Sleep(100 * time.Second)
            tcpConn.Close()
            fmt.Println("tcp end")
        }
    }()

    wg.Wait()

}

io.Copy 將回傳「splice: 連線逾時」

如果我啟用 keepalive

sudo sysctl -w net.ipv4.tcp_keepalive_probes=3

io.Copy就可以了

我嘗試tcp客戶端以1秒的間隔發送資料包並停用tcp_keepalive,也可以。

我寫了另一個程式碼來取代 io.Copy

            buf := make([]byte, 10)
            for {
                conn.SetReadDeadline(time.Now().Add(30 * time.Second))
                n, err := conn.Read(buf)
                if err != nil {
                    fmt.Println("read fail: ", err.Error())
                    break
                }

                _, err = conn.Write(buf[:n])
                if err != nil {
                    fmt.Println("write fail: ", err.Error())
                    break
                }
            }
            conn.Close()

15秒後讀取逾時,30秒不工作

為什麼會這樣,太奇怪了?

解決方法

答案: 當「kernel tcp stack」使用keepalive偵測網路時,當「kernel tcp stack」需要發送keepalive時,net.ipv4.tcp_keepalive_probes = 0將會逾時

以上是停用 keepalive 時 io.Copy 逾時的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:stackoverflow.com。如有侵權,請聯絡admin@php.cn刪除