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中文網其他相關文章!