套接字繫結失敗:已被佔用問題
阿新 • • 發佈:2018-12-31
bind API 函式來繫結一個地址(一個介面和一個埠)到一個套接字端點。可以在伺服器設定中使用這個函式,以便限制可能有連線到來的介面。也可以在客戶端設定中使用 這個函式,以便限制應當供出去的連線所使用的介面。bind 最常見的用法是關聯埠號和伺服器,並使用萬用字元地址(INADDR_ANY),它允許任何介面為到來的連線所使用。
bind 普遍遭遇的問題是試圖繫結一個已經在使用的埠。該隱患是也許沒有活動的套接字存在,但仍然禁止繫結埠(bind 返回 EADDRINUSE),它由 TCP 套接字狀態 TIME_WAIT 引起。該狀態在套接字關閉後約保留 2 到 4 分鐘。在 TIME_WAIT 狀態退出之後,套接字被刪除,該地址才能被重新繫結而不出問題。
等待 TIME_WAIT 結束可能是令人惱火的一件事,特別是如果您正在開發一個套接字伺服器,就需要停止伺服器來做一些改動,然後重啟。幸運的是,有方法可以避開 TIME_WAIT 狀態。可以給套接字應用 SO_REUSEADDR 套接字選項,以便埠可以馬上重用。
解決: 使用套介面函式setsockopt()開啟SO_REUSEADDR 選項
第三個引數為1,允許地址重用
第三個引數為0,禁止地址重用
- int sock, ret, on;
- struct sockaddr_in servaddr;
- /* Create a new stream (TCP) socket */
- sock = socket( AF_INET, SOCK_STREAM, 0 ):
- /* Enable address reuse */
- on = 1;
- ret = setsockopt( sock, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on) );
- /* Allow connections to port 8080 from any available interface */
- memset( &servaddr, 0, sizeof(servaddr) );
- servaddr.sin_family = AF_INET;
- servaddr.sin_addr.s_addr = htonl( INADDR_ANY );
- servaddr.sin_port = htons( 45000 );
- /* Bind to the address (interface/port) */
- ret = bind( sock, (struct sockaddr *)&servaddr, sizeof(servaddr) );
在應用了 SO_REUSEADDR 選項之後,bind API 函式將允許地址的立即重用。