Linux 套接字编程中 bind 错误:bind fail:Address already in use 解决方法

原创 2016年08月31日 14:17:56

最近在学Linux 网络编程,调试TCP并发服务器时遇到一个问题,当我连接上一个或多个客户端后,用 CTRL+C 关闭进程后,重新打开进程就发生错误了:bind fail:Address already in use

地址被占用???

开始我以为是套接字描述符未关闭,添加代码在发生错误时关闭掉套接字描述符还是不行。

上网查了才发现是原来是套接字状态未配置,IBM官网上有较为详细的解释Linux 套接字编程中的 5 个隐患

下面截取文中一段相关内容:

隐患 3.地址使用错误(EADDRINUSE)

您可以使用 bind API 函数来绑定一个地址(一个接口和一个端口)到一个套接字端点。可以在服务器设置中使用这个函数,以便限制可能有连接到来的接口。也可以在客户端设置中使用这个函数,以便限制应当供出去的连接所使用的接口。bind 最常见的用法是关联端口号和服务器,并使用通配符地址(INADDR_ANY),它允许任何接口为到来的连接所使用。

bind 普遍遭遇的问题是试图绑定一个已经在使用的端口。该陷阱是也许没有活动的套接字存在,但仍然禁止绑定端口(bind 返回EADDRINUSE),它由 TCP 套接字状态 TIME_WAIT 引起。该状态在套接字关闭后约保留 2 到 4 分钟。在 TIME_WAIT 状态退出之后,套接字被删除,该地址才能被重新绑定而不出问题。

等待 TIME_WAIT 结束可能是令人恼火的一件事,特别是如果您正在开发一个套接字服务器,就需要停止服务器来做一些改动,然后重启。幸运的是,有方法可以避开 TIME_WAIT 状态。可以给套接字应用 SO_REUSEADDR 套接字选项,以便端口可以马上重用。


原来是TCP 套接字状态 TIME_WAIT 引起的,解决方法就是用 setsockopt 函数对套接字状态进行配置:

int iSockOptVal = 1;
if (setsockopt(iSockFd, SOL_SOCKET, SO_REUSEADDR, &iSockOptVal, sizeof(iSockOptVal)) == -1) {
	perror("setsockopt fail");
	close(iSockFd);
	exit(EXIT_FAILURE);
}

上述代码中,函数原型为 setsockopt (int s, int level, int optname, const void *optval, socklen_t optlen),

参数 s为 socket 描述符;

level 代表预设置的网络层,一般设置为 SOLSOCKET 以存取 socket 层;

optname 代表预设置的选项,设为 SO_REUSEADDR 表示允许在 bind() 时本地地址可重复复用;

optval 代表预设置的值的指针,在这里传入1表示允许地址重复,传入0表示不允许地址重复; 

optlen 则为 optval 的长度。

关于 setsockopt 函数的配置参数的一些解析及应用可参考这篇文章:setsockopt 用法详解

大笑

附上我的TCP并发服务器测试代码:

#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include 
#include 
#include 

static void *clientHandle(void *arg);

int main(int argc, char *argv[]) 
{ 
	int iNetPort = 0;
	int iSockFd = 0;
	int iClientFd = 0;
	int iSerAddrLen = 0;
	int iSockOptVal = 1;
	pthread_t iThreadId = 0;
	struct sockaddr_in SERVERADDR;
	/*
	*  输入参数检查
	*/
	if (argc != 2) {
		printf("input one arg\r\n");
		exit(EXIT_FAILURE);
	}
	iNetPort = atoi(argv[1]);
	
	/*
	*  建立套接字描述符
	*/
	if ((iSockFd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
		perror("socket fail");
		exit(EXIT_FAILURE);
	}
	/*
	*  设置套接字状态
	*/
	if (setsockopt(iSockFd, SOL_SOCKET, SO_REUSEADDR, &iSockOptVal, sizeof(iSockOptVal)) == -1) {
		perror("setsockopt fail");
		close(iSockFd);
		exit(EXIT_FAILURE);
	}
	/*
	* 绑定服务器与套接字
	*/
	bzero(&SERVERADDR, sizeof(SERVERADDR));
	SERVERADDR.sin_family = AF_INET;
	SERVERADDR.sin_port = htons(iNetPort);
	SERVERADDR.sin_addr.s_addr = htonl(INADDR_ANY);
	if (bind(iSockFd, (struct sockaddr *)&SERVERADDR, sizeof(SERVERADDR)) == -1) {
		perror("bind fail");
		close(iSockFd);
		exit(EXIT_FAILURE);
	}
	
	/*
	*  监听指定端口,最大连接5个客户端
	*/
	if (listen(iSockFd, 5) == -1) {
		perror("listen fail");
		close(iSockFd);
		exit(EXIT_FAILURE);
	}
	
	/*
	*  为每个连接的客户端建立一个线程
	*/
	while(1) {
		iSerAddrLen = sizeof(SERVERADDR);
		if ((iClientFd = accept(iSockFd, (struct sockaddr *)&SERVERADDR, &iSerAddrLen)) == -1) {
			if (errno == EINTR) {
				close(iSockFd);
				continue;
			}
			else {
				perror("accept fail");
				close(iSockFd);
				exit(EXIT_FAILURE);
			}
		}
		/*
		*  打印客户地址
		*/
		printf("Client IP:%s\r\n",inet_ntoa(SERVERADDR.sin_addr));
		printf("Client PORT:%d\r\n",ntohs(SERVERADDR.sin_port));
		
		/*
		*  新建一个线程
		*/
		if (pthread_create(&iThreadId, NULL, clientHandle, &iClientFd) == -1) {
			perror("pthread_create fail");
			close(iSockFd);
			close(iClientFd);
			exit(EXIT_FAILURE);
		}
	}
	close(iSockFd);
	close(iClientFd);
	exit(EXIT_SUCCESS);
} 

static void *clientHandle(void *arg)
{
	int iClientFd = *(int *)arg;
	int iReadByteSize = 0;
	char cRcvSndBuf[100];
	time_t tTime;
	
	printf("Client Fd:%d\r\n",iClientFd);
	while (1) {
		/*
		*  接受客户端信息
		*/
		memset(cRcvSndBuf, 0, sizeof(cRcvSndBuf));
		if((iReadByteSize = read(iClientFd, cRcvSndBuf, sizeof(cRcvSndBuf))) == -1) {
			perror("read fail");
			close(iClientFd);
			return 0;
		}
		else if (iReadByteSize == 0) {
			printf("Client not connect");
			close(iClientFd);
			return 0;
		}
		
		if (strncmp(cRcvSndBuf, "end", 3) == 0) {
			printf("thread close:%d\r\n", iClientFd);
			close(iClientFd);
			return 0;
		}
		else if(strncmp(cRcvSndBuf, "time", 4) == 0) {
			tTime = time(NULL);
			sprintf(cRcvSndBuf, "TIME:%s", ctime(&tTime));
		}
		else {
			cRcvSndBuf[iReadByteSize] = '\n';
			cRcvSndBuf[iReadByteSize+1] = '\0';
		}
		write(iClientFd, cRcvSndBuf, strlen(cRcvSndBuf));
	}
	return 0;
}

bind:address already in use的深刻教训以及解决办法

今天在linux下,编写了一个简单的回射客户/服务器(就是客户机从控制台标准输入并发送数据,服务端接受数据,但是不对数据进行处理,然后将数据返回,交由客户机标准输出),然后遇到了一些问题,郁闷了好长时...
  • msdnwolaile
  • msdnwolaile
  • 2016年02月26日 09:24
  • 14826

perror()打印“ Address already in use ”的解决方法,key words:errno perror bind setsockopt

代码如下:if( bind(local_sock_fd,(struct sockaddr*)(&local_sock_addr),sizeof(struct sockaddr)) == -1){   ...
  • dos5gw
  • dos5gw
  • 2011年03月14日 10:07
  • 1106

bind出现Address already in use的原因及解决方法

每次修改了源代码并再次编译运行时,常遇到下面的地使用错误: Cann't bind server socket ! : Address already in use 虽然用Ctrl+C强制结束了进程,...
  • caomiao2006
  • caomiao2006
  • 2016年05月14日 22:05
  • 1544

Address already in use JVM_Bind8080类似问题解决方法

出现如下问题是因为8080端口被占用,出现这种情况有可能是因为安装了两个tomcat服务器(我出现的问题就是这样的)。有可能是tomcat和oracle都被安装了。 1.严重: Error init...
  • Nausicaa12
  • Nausicaa12
  • 2012年10月01日 23:12
  • 762

Address already in use: JVM_Bind<null>:8080错误的解决办法

严重:Error initializing endpoint java.net.BindException: Address already in use: JVM_Bind:8080 at or...
  • mixiuali
  • mixiuali
  • 2012年09月15日 10:58
  • 75261

Linux下Socket编程的端口问题( Bind(): Address already in use )

Linux下Socket编程的端口问题( Bind(): Address already in use ) 转载请注明 本文引用地址 http://blog.csdn.net/xl_xunzhao/...
  • dragon101788
  • dragon101788
  • 2012年08月24日 11:40
  • 1969

Linux下Socket编程的端口问题( Bind error: Address already in use )

在进行linux网络编程时,每次修改了源代码并再次编译运行时,常遇到下面的地使用错误: Bind error: Address already in use 虽然用Ctrl+C强制结束了进程,但错...
  • bbs598598
  • bbs598598
  • 2012年05月01日 10:12
  • 740

socket编程注意的地方---bind socket error: Address already in use

在进行linux网络编程时,每次修改了源代码并再次编译运行时,常遇到下面的地使用错误: Bind error: Address already in use 虽然用Ctrl...
  • chenxun2009
  • chenxun2009
  • 2015年10月30日 21:00
  • 2217

启动程序端口被占用Address already in use: bind解决方案

例如我们web服务器8080端口被占用:Address already in use: bind 有两种解决方案:1、把占用8080端口的程序终止掉。2、修改tomcat配置文件里的默认端口号 1...
  • yels303
  • yels303
  • 2016年03月15日 21:36
  • 2934

解决nginx: [emerg] bind() to [::]:80 failed (98: Address already in use)

首选是安装nginx,在./configure出错的情况下,查看错误的报告,如果是缺少一些必要的库,安装这些库,可能的有ssl pcre(这个库我是使用源码安装的,在线安装没有成功,提示看不到这个库)...
  • yusiguyuan
  • yusiguyuan
  • 2014年03月05日 20:04
  • 15434
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Linux 套接字编程中 bind 错误:bind fail:Address already in use 解决方法
举报原因:
原因补充:

(最多只允许输入30个字)