Address already in use的解決方法


當客戶端保持着與服務器端的連接,這時服務器端斷開,再開啟服務器時會出現: Address already in usr
可以用netstat -anp | more 可以看到客戶端還保持着與服務器的連接(還在使用服務器bind的端口)。這是由於client沒有執行close,連接還會等待client的FIN包一段時間。解決方法是使用setsockopt,使得socket可以被重用,是最常用的服務器編程要點。具體的做法為是,在socket調用和bind調用之間加上一段對socket的設置:
   int opt = 1;
   setsockopt(socket_fd,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt));
附 :setsockopt的用法。
 
setsockopt(設置socket狀態) 
相關函數 
getsockopt
表頭文件 
#include;
#include;
定義函數 
int setsockopt(int s,int level,int optname,const void * optval,,socklen_toptlen);
函數說明 
setsockopt()用來設置參數s所指定的socket狀態。參數level代表欲設置的網絡層,一般設成SOL_SOCKET以存取socket層。參數optname代表欲設置的選項,有下列幾種數值:
SO_DEBUG 打開或關閉排錯模式
SO_REUSEADDR 允許在bind()過程中本地地址可重復使用
SO_TYPE 返回socket形態。
SO_ERROR 返回socket已發生的錯誤原因
SO_DONTROUTE 送出的數據包不要利用路由設備來傳輸。
SO_BROADCAST 使用廣播方式傳送
SO_SNDBUF 設置送出的暫存區大小
SO_RCVBUF 設置接收的暫存區大小
SO_KEEPALIVE 定期確定連線是否已終止。
SO_OOBINLINE 當接收到OOB 數據時會馬上送至標准輸入設備
SO_LINGER 確保數據安全且可靠的傳送出去。
參數 
optval代表欲設置的值,參數optlen則為optval的長度。
返回值 
成功則返回0,若有錯誤則返回-1,錯誤原因存於errno。
附加說明 
EBADF 參數s並非合法的socket處理代碼
ENOTSOCK 參數s為一文件描述詞,非socket
ENOPROTOOPT 參數optname指定的選項不正確。
EFAULT 參數optval指針指向無法存取的內存空間

我也是剛開始學習socket網絡編程,就在同學那搞到一個完整版的socket,我在centos上的eclipse下進行了編譯,就是總出現Address already in use,后來發現了一篇博客,就是下文,試了試還是沒用,結果就想一定是進程沒有殺干凈,所以就在eclipse下file里找到了restart,試了試果然好使!不過本人覺得下面一篇文章還是很好的!

 

 

 

 

很多socket編程的初學者可能會遇到這樣的問題:如果先ctrl+c結束服務器端程序的話,再次啟動服務器就會出現Address already in use這個錯誤,或者你的程序在正常關閉服務器端socket后還是有這個問題。正如下面的這段簡單的socket程序。

 

server.c

 

[c-sharp]  view plain copy
 
  1. #include <sys/types.h>   
  2. #include <sys/socket.h>   
  3. #include <stdio.h>   
  4. #include <netinet/in.h>   
  5. #include <arpa/inet.h>  
  6. #include <unistd.h>  
  7. #include <stdlib.h>  
  8.  
  9. #define BUFFER_SIZE 40   
  10.   
  11. int main()   
  12. {         
  13.     char buf[BUFFER_SIZE];  
  14.     int server_sockfd, client_sockfd;   
  15.         int sin_size=sizeof(struct sockaddr_in);   
  16.     struct sockaddr_in server_address;   
  17.     struct sockaddr_in client_address;   
  18.     memset(&server_address,0,sizeof(server_address));  
  19.     server_address.sin_family = AF_INET;   
  20.     server_address.sin_addr.s_addr = INADDR_ANY;   
  21.     server_address.sin_port = htons(12000);   
  22.     // 建立服務器端socket    
  23.     if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)  
  24.     {  
  25.         perror("server_sockfd creation failed");  
  26.         exit(EXIT_FAILURE);  
  27.     }  
  28.     // 將套接字綁定到服務器的網絡地址上    
  29.     if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)  
  30.     {  
  31.         perror("server socket bind failed");  
  32.         exit(EXIT_FAILURE);  
  33.     }  
  34.     // 建立監聽隊列   
  35.     listen(server_sockfd,5);  
  36.     // 等待客戶端連接請求到達   
  37.     client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);  
  38.     if(client_sockfd<0)  
  39.     {  
  40.         perror("accept client socket failed");  
  41.         exit(EXIT_FAILURE);  
  42.     }  
  43.     // 接收客戶端數據   
  44.     if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)  
  45.     {  
  46.         perror("recv client data failed");  
  47.         exit(EXIT_FAILURE);  
  48.     }  
  49.     printf("receive from client:%s/n",buf);  
  50.     // 發送數據到客戶端   
  51.     if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)  
  52.     {  
  53.         perror("send failed");  
  54.         exit(EXIT_FAILURE);  
  55.     }  
  56.     close(client_sockfd);  
  57.     close(server_sockfd);  
  58.     exit(EXIT_SUCCESS);  
  59. }  
[c-sharp]  view plain  copy
 
  1. #include <sys/types.h>   
  2. #include <sys/socket.h>   
  3. #include <stdio.h>   
  4. #include <netinet/in.h>   
  5. #include <arpa/inet.h>  
  6. #include <unistd.h>  
  7. #include <stdlib.h>  
  8.  
  9. #define BUFFER_SIZE 40  
  10.   
  11. int main()   
  12. {         
  13.     char buf[BUFFER_SIZE];  
  14.     int server_sockfd, client_sockfd;   
  15.         int sin_size=sizeof(struct sockaddr_in);   
  16.     struct sockaddr_in server_address;   
  17.     struct sockaddr_in client_address;   
  18.     memset(&server_address,0,sizeof(server_address));  
  19.     server_address.sin_family = AF_INET;   
  20.     server_address.sin_addr.s_addr = INADDR_ANY;   
  21.     server_address.sin_port = htons(12000);   
  22.     // 建立服務器端socket   
  23.     if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)  
  24.     {  
  25.         perror("server_sockfd creation failed");  
  26.         exit(EXIT_FAILURE);  
  27.     }  
  28.     // 將套接字綁定到服務器的網絡地址上   
  29.     if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)  
  30.     {  
  31.         perror("server socket bind failed");  
  32.         exit(EXIT_FAILURE);  
  33.     }  
  34.     // 建立監聽隊列  
  35.     listen(server_sockfd,5);  
  36.     // 等待客戶端連接請求到達  
  37.     client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);  
  38.     if(client_sockfd<0)  
  39.     {  
  40.         perror("accept client socket failed");  
  41.         exit(EXIT_FAILURE);  
  42.     }  
  43.     // 接收客戶端數據  
  44.     if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)  
  45.     {  
  46.         perror("recv client data failed");  
  47.         exit(EXIT_FAILURE);  
  48.     }  
  49.     printf("receive from client:%s/n",buf);  
  50.     // 發送數據到客戶端  
  51.     if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)  
  52.     {  
  53.         perror("send failed");  
  54.         exit(EXIT_FAILURE);  
  55.     }  
  56.     close(client_sockfd);  
  57.     close(server_sockfd);  
  58.     exit(EXIT_SUCCESS);  
  59. }  

 

 

 

client.c

 

[c-sharp]  view plain copy
 
  1. #include <sys/types.h>   
  2. #include <sys/socket.h>   
  3. #include <stdio.h>   
  4. #include <netinet/in.h>                                                 
  5. #include <arpa/inet.h>   
  6. #include <unistd.h>   
  7. #include <stdlib.h>  
  8.  
  9. #define BUFFER_SIZE 40   
  10.   
  11. int main()   
  12. {   
  13.     char buf[BUFFER_SIZE];  
  14.     int client_sockfd;   
  15.     int len;   
  16.     struct sockaddr_in address;// 服務器端網絡地址結構體                                             
  17.      int result;   
  18.     client_sockfd = socket(AF_INET, SOCK_STREAM, 0);// 建立客戶端socket                                 
  19.     address.sin_family = AF_INET;   
  20.     address.sin_addr.s_addr = inet_addr("127.0.0.1");               
  21.     address.sin_port = htons(12000);   
  22.     len = sizeof(address);  
  23.     // 與遠程服務器建立連接   
  24.     result = connect(client_sockfd, (struct sockaddr *)&address, len);   
  25.     if(result<0)   
  26.     {   
  27.          perror("connect failed");   
  28.          exit(EXIT_FAILURE);   
  29.     }   
  30.     printf("Please input the message:");  
  31.     scanf("%s",buf);  
  32.     send(client_sockfd,buf,BUFFER_SIZE,0);  
  33.     recv(client_sockfd,buf,BUFFER_SIZE,0);  
  34.     printf("receive data from server: %s/n",buf);  
  35.     close(client_sockfd);   
  36.     return 0;   
  37. }  
[c-sharp]  view plain  copy
 
  1. #include <sys/types.h>   
  2. #include <sys/socket.h>   
  3. #include <stdio.h>   
  4. #include <netinet/in.h>                                                  
  5. #include <arpa/inet.h>   
  6. #include <unistd.h>   
  7. #include <stdlib.h>  
  8.  
  9. #define BUFFER_SIZE 40  
  10.   
  11. int main()   
  12. {   
  13.     char buf[BUFFER_SIZE];  
  14.     int client_sockfd;   
  15.     int len;   
  16.     struct sockaddr_in address;// 服務器端網絡地址結構體                                             
  17.      int result;   
  18.     client_sockfd = socket(AF_INET, SOCK_STREAM, 0);// 建立客戶端socket                                 
  19.     address.sin_family = AF_INET;   
  20.     address.sin_addr.s_addr = inet_addr("127.0.0.1");               
  21.     address.sin_port = htons(12000);   
  22.     len = sizeof(address);  
  23.     // 與遠程服務器建立連接  
  24.     result = connect(client_sockfd, (struct sockaddr *)&address, len);   
  25.     if(result<0)   
  26.     {   
  27.          perror("connect failed");   
  28.          exit(EXIT_FAILURE);   
  29.     }   
  30.     printf("Please input the message:");  
  31.     scanf("%s",buf);  
  32.     send(client_sockfd,buf,BUFFER_SIZE,0);  
  33.     recv(client_sockfd,buf,BUFFER_SIZE,0);  
  34.     printf("receive data from server: %s/n",buf);  
  35.     close(client_sockfd);   
  36.     return 0;   
  37. }  

 

 

 

      在成功的運行了第一次之后,當你再次啟動服務器端程序時,./server就變得邪惡起來,在bind()這個函數中居然出現了Address already in use這個錯誤。

                                  

      然后你開始迷惑了,難道是忘記將socket給關閉了,或是關閉socket的順序不對?經過種種猜測與試驗,你發現問題毫無進展......過了一會,當你再次抱着試試看的態度重新在Linux的“黑色終端”中輸入./server時,程序居然運行了,什么情況?究其原因,是socket選項在搗鬼。下面是IBM官網上對這一情況的具體解釋,參見http://www.ibm.com/developerworks/cn/linux/l-sockpit/

      bind 普遍遭遇的問題是試圖綁定一個已經在使用的端口。該陷阱是也許沒有活動的套接字存在,但仍然禁止綁定端口(bind 返回 EADDRINUSE),它由 TCP 套接字狀態 TIME_WAIT 引起。該狀態在套接字關閉后約保留 2 到 4 分鍾。在 TIME_WAIT 狀態退出之后,套接字被刪除,該地址才能被重新綁定而不出問題。

等待 TIME_WAIT 結束可能是令人惱火的一件事,特別是如果您正在開發一個套接字服務器,就需要停止服務器來做一些改動,然后重啟。幸運的是,有方法可以避開 TIME_WAIT 狀態。可以給套接字應用 SO_REUSEADDR 套接字選項,以便端口可以馬上重用。

考慮清單 3 的例子。在綁定地址之前,我以 SO_REUSEADDR 選項調用 setsockopt。為了允許地址重用,我設置整型參數(on)為 1 (不然,可以設為 0 來禁止地址重用)。

      按照IBM的做法,我重新改寫了server.c的代碼。

server.c

 

[c-sharp]  view plain copy
 
  1. #include <sys/types.h>   
  2. #include <sys/socket.h>   
  3. #include <stdio.h>   
  4. #include <netinet/in.h>   
  5. #include <arpa/inet.h>  
  6. #include <unistd.h>  
  7. #include <stdlib.h>  
  8.  
  9. #define BUFFER_SIZE 40   
  10.   
  11. int main()   
  12. {         
  13.     char buf[BUFFER_SIZE];  
  14.     int server_sockfd, client_sockfd;   
  15.         int sin_size=sizeof(struct sockaddr_in);   
  16.     struct sockaddr_in server_address;   
  17.     struct sockaddr_in client_address;   
  18.     memset(&server_address,0,sizeof(server_address));  
  19.     server_address.sin_family = AF_INET;   
  20.     server_address.sin_addr.s_addr = INADDR_ANY;   
  21.     server_address.sin_port = htons(12000);   
  22.     // 建立服務器端socket    
  23.     if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)  
  24.     {  
  25.         perror("server_sockfd creation failed");  
  26.         exit(EXIT_FAILURE);  
  27.     }  
  28.     // 設置套接字選項避免地址使用錯誤   
  29.     int on=1;  
  30.     if((setsockopt(server_sockfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on)))<0)  
  31.     {  
  32.         perror("setsockopt failed");  
  33.         exit(EXIT_FAILURE);  
  34.     }  
  35.     // 將套接字綁定到服務器的網絡地址上    
  36.     if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)  
  37.     {  
  38.         perror("server socket bind failed");  
  39.         exit(EXIT_FAILURE);  
  40.     }  
  41.     // 建立監聽隊列   
  42.     listen(server_sockfd,5);  
  43.     // 等待客戶端連接請求到達   
  44.     client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);  
  45.     if(client_sockfd<0)  
  46.     {  
  47.         perror("accept client socket failed");  
  48.         exit(EXIT_FAILURE);  
  49.     }  
  50.     // 接收客戶端數據   
  51.     if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)  
  52.     {  
  53.         perror("recv client data failed");  
  54.         exit(EXIT_FAILURE);  
  55.     }  
  56.     printf("receive from client:%s/n",buf);  
  57.     // 發送數據到客戶端   
  58.     if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)  
  59.     {  
  60.         perror("send failed");  
  61.         exit(EXIT_FAILURE);  
  62.     }  
  63.     close(client_sockfd);  
  64.     close(server_sockfd);  
  65.     exit(EXIT_SUCCESS);  
  66. }  
[c-sharp]  view plain  copy
 
  1. #include <sys/types.h>   
  2. #include <sys/socket.h>   
  3. #include <stdio.h>   
  4. #include <netinet/in.h>   
  5. #include <arpa/inet.h>  
  6. #include <unistd.h>  
  7. #include <stdlib.h>  
  8.  
  9. #define BUFFER_SIZE 40  
  10.   
  11. int main()   
  12. {         
  13.     char buf[BUFFER_SIZE];  
  14.     int server_sockfd, client_sockfd;   
  15.         int sin_size=sizeof(struct sockaddr_in);   
  16.     struct sockaddr_in server_address;   
  17.     struct sockaddr_in client_address;   
  18.     memset(&server_address,0,sizeof(server_address));  
  19.     server_address.sin_family = AF_INET;   
  20.     server_address.sin_addr.s_addr = INADDR_ANY;   
  21.     server_address.sin_port = htons(12000);   
  22.     // 建立服務器端socket   
  23.     if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)  
  24.     {  
  25.         perror("server_sockfd creation failed");  
  26.         exit(EXIT_FAILURE);  
  27.     }  
  28.     // 設置套接字選項避免地址使用錯誤  
  29.     int on=1;  
  30.     if((setsockopt(server_sockfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on)))<0)  
  31.     {  
  32.         perror("setsockopt failed");  
  33.         exit(EXIT_FAILURE);  
  34.     }  
  35.     // 將套接字綁定到服務器的網絡地址上   
  36.     if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)  
  37.     {  
  38.         perror("server socket bind failed");  
  39.         exit(EXIT_FAILURE);  
  40.     }  
  41.     // 建立監聽隊列  
  42.     listen(server_sockfd,5);  
  43.     // 等待客戶端連接請求到達  
  44.     client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);  
  45.     if(client_sockfd<0)  
  46.     {  
  47.         perror("accept client socket failed");  
  48.         exit(EXIT_FAILURE);  
  49.     }  
  50.     // 接收客戶端數據  
  51.     if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)  
  52.     {  
  53.         perror("recv client data failed");  
  54.         exit(EXIT_FAILURE);  
  55.     }  
  56.     printf("receive from client:%s/n",buf);  
  57.     // 發送數據到客戶端  
  58.     if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)  
  59.     {  
  60.         perror("send failed");  
  61.         exit(EXIT_FAILURE);  
  62.     }  
  63.     close(client_sockfd);  
  64.     close(server_sockfd);  
  65.     exit(EXIT_SUCCESS);  
  66. }  

 

 

       這次,讓我們再次反復的啟動服務器,盡情的在“黑窗戶”里面輸入./server ./server ./server ......服務器的程序好像突然間變乖了,呵呵,童鞋們,為自己的成就慶祝吧!!!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM