socket編程小問題:地址已經被使用——Address already in use


轉自:http://blog.csdn.net/a_tu_/article/details/44625117

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

server.cpp

 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.cpp

 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

 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