bind error : Cannot assign requested address
代码:
#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<stdlib.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<string.h>
static void usage(const char* proc)
{
printf("Usage: %s [local_ip] [local_port]",proc);
}
int startup(const char* ip,int port)
{
int sock = socket(AF_INET,SOCK_STREAM,0);
if(sock<0)
{
perror("socket");
exit(2);
}
int on = 1;
int s = setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on));
if(s<0){
perror("setsockopt");
exit(6);
}
struct sockaddr_in local;
local.sin_family = AF_INET;
local.sin_port = htons(port);
local.sin_addr.s_addr = inet_addr(ip);
if(bind(sock,(struct sockaddr*) &local,sizeof(local))<0){
perror("bind");
exit(3);
}
if(listen(sock,10)<0)
{
perror("listen");
exit(4);
}
return sock;
}
//Version 1
//./tcp_servet 127.0.0.1 8080
int main(int argc,char* argv[])
{
if(argc!=3){
usage(argv[0]);
return 1;
}
int listen_sock = startup(argv[1],atoi(argv[2]));//create socket
int new_sock;
while(1){
struct sockaddr_in client;
socklen_t len = sizeof(client);
new_sock = accept(listen_sock,(struct sockaddr*)&client ,&len);
if(new_sock<0)
{
perror("accept");
exit(5);
continue;
}
printf("get an new client: %s, %d\n",inet_ntoa(client.sin_addr),ntohs(client.sin_port));
break;
}
char buf[1024];
while(1){
int r = read(new_sock,buf,sizeof(buf)-1);
if(r<0){
perror("read");
close(new_sock);
break;
}else if(r==0){//clinet close
close(new_sock);
printf("client qiuit...\n");
break;
}
buf[r] = 0;
printf("clinet: %s\n",buf);
write(new_sock,buf,strlen(buf));
}
return 0;
}
错误情况如图:
原因:
客户端频繁的连服务器,由于每次连接都在很短的时间内结束,导致很多的TIME_WAIT,以至于用光了可用的端 口号,所以新的连接没办法绑定端口,即“Cannot assign requested address”。是客户端的问题不是服务器端的问题。通过netstat,的确看到很多TIME_WAIT状态的连接。
可见问题不在于server端,而是client端频繁建立连接,而端口释放较慢,导致建立新连接时无可用端口。
解决办法:
执行命令修改如下2个内核参数 (需要root权限)
- sysctl -w net.ipv4.tcp_timestamps=1 开启对于TCP时间戳的支持,若该项设置为0,则下面一项设置不起作用
- sysctl -w net.ipv4.tcp_tw_recycle=1 表示开启TCP连接中TIME-WAIT sockets的快速回收