Golang 端口復(fù)用測(cè)試的實(shí)現(xiàn)
先給出結(jié)論:
同一個(gè)進(jìn)程,使用一個(gè)端口,然后連接關(guān)閉,大約需要30s后才可再次使用這個(gè)端口。
測(cè)試
首先使用端口9001連接服務(wù)端,發(fā)送數(shù)據(jù),然后關(guān)閉連接,接著再次使用端口9001連接服務(wù)端,如果連接失敗,間隔15s后,再次嘗試,最多嘗試3次,。
client
package main import ( "bufio" "fmt" "net" "os" "time" ) func DialCustom(network, address string, timeout time.Duration, localIP []byte, localPort int)(net.Conn,error) { netAddr := &net.TCPAddr{Port:localPort} if len(localIP) != 0 { netAddr.IP = localIP } fmt.Println("netAddr:", netAddr) d := net.Dialer{Timeout: timeout, LocalAddr: netAddr} return d.Dial(network, address) } func getOneConn() { serverAddr := "127.0.0.1:8080" // 172.28.172.180 //localIP := []byte{0xAC, 0x1C, 0xAC, 0xB4} // IP localIP := []byte{} // any IP localPort := 9001 var conn net.Conn var err error for i:=0;i<3;i++{ conn, err = DialCustom("tcp", serverAddr, time.Second*10, localIP,localPort) if err != nil { fmt.Println("dial failed:", err) if i == 2 { os.Exit(1) } time.Sleep(15*time.Second) } else { break } } defer conn.Close() buffer := make([]byte, 512) reader := bufio.NewReader(conn) n, err2 := reader.Read(buffer) if err2 != nil { fmt.Println("Read failed:", err2) return } fmt.Println("count:", n, "msg:", string(buffer)) } func main() { getOneConn() fmt.Println("=========================") getOneConn() fmt.Println("=========================") select{} }
server
package main import ( "fmt" "net" "log" ) func main() { addr := "0.0.0.0:8080" tcpAddr, err := net.ResolveTCPAddr("tcp",addr) if err != nil { log.Fatalf("net.ResovleTCPAddr fail:%s", addr) } listener, err := net.ListenTCP("tcp", tcpAddr) if err != nil { log.Fatalf("listen %s fail: %s", addr, err) } else { log.Println("rpc listening", addr) } for { conn, err := listener.Accept() if err != nil { log.Println("listener.Accept error:", err) continue } go handleConnection(conn) } } func handleConnection(conn net.Conn) { //defer conn.Close() var buffer []byte = []byte("You are welcome. I'm server.") n, err := conn.Write(buffer) if err != nil { fmt.Println("Write error:", err) } fmt.Println("send:", n) fmt.Println("connetion end") }
output
client輸出:
$ ./client
netAddr: :9001
count: 28 msg: You are welcome. I'm server.
=========================
netAddr: :9001
dial failed: dial tcp :9001->127.0.0.1:8080: bind: address already in use
netAddr: :9001
dial failed: dial tcp :9001->127.0.0.1:8080: bind: address already in usenetAddr: :9001
count: 28 msg: You are welcome. I'm server.
=========================
經(jīng)過3次重試,30s后,才可以重新使用同一個(gè)端口9001進(jìn)行連接。也就是同一個(gè)進(jìn)程的情況狂下,一個(gè)連接關(guān)閉后,端口大約30s后才可以被使用。
到此這篇關(guān)于Golang 端口復(fù)用測(cè)試的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Golang 端口復(fù)用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語(yǔ)言中init函數(shù)和defer延遲調(diào)用關(guān)鍵詞詳解
這篇文章主要介紹了Go語(yǔ)言中init函數(shù)和defer延遲調(diào)用關(guān)鍵詞,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-03-03Golang中的[]byte與16進(jìn)制(String)之間的轉(zhuǎn)換方式
這篇文章主要介紹了Golang中的[]byte與16進(jìn)制(String)之間的轉(zhuǎn)換方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11Go語(yǔ)言使用defer+recover解決panic導(dǎo)致程序崩潰的問題
如果協(xié)程出現(xiàn)了panic,就會(huì)造成程序的崩潰,這時(shí)可以在goroutine中使用recover來捕獲panic,進(jìn)行處理,本文就詳細(xì)的介紹一下,感興趣的可以了解一下2021-09-09Go項(xiàng)目與Docker結(jié)合實(shí)現(xiàn)高效部署深入探究
在現(xiàn)代軟件開發(fā)中,使用Docker部署應(yīng)用程序已經(jīng)成為一種標(biāo)準(zhǔn)實(shí)踐,本文將深入探討如何將Go項(xiàng)目與Docker結(jié)合,實(shí)現(xiàn)高效、可靠的部署過程,通過詳細(xì)的步驟和豐富的示例,你將能夠迅速掌握這一流程2023-12-12Go語(yǔ)言開發(fā)kube-scheduler整體架構(gòu)深度剖析
這篇文章主要為大家介紹了Go語(yǔ)言開發(fā)kube-scheduler整體架構(gòu)深度剖析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04Goland 關(guān)閉自動(dòng)移除未使用的包操作
這篇文章主要介紹了Goland 關(guān)閉自動(dòng)移除未使用的包操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-12-12源碼剖析Golang中map擴(kuò)容底層的實(shí)現(xiàn)
之前的文章詳細(xì)介紹過Go切片和map的基本使用,以及切片的擴(kuò)容機(jī)制。本文針對(duì)map的擴(kuò)容,會(huì)從源碼的角度全面的剖析一下map擴(kuò)容的底層實(shí)現(xiàn),需要的可以參考一下2023-03-03Go 語(yǔ)言結(jié)構(gòu)實(shí)例分析
在本篇文章里小編給大家整理的是一篇關(guān)于Go 語(yǔ)言結(jié)構(gòu)實(shí)例分析的相關(guān)知識(shí)點(diǎn),有興趣的朋友們可以學(xué)習(xí)下。2021-07-07