专栏首页Golang语言社区socket编程小问题:地址已经被使用——Address already in use

socket编程小问题:地址已经被使用——Address already in use

很多socket编程的初学者可能会遇到这样的问题:如果先ctrl+c结束服务器端程序的话,再次启动服务器就会出现Address already in use这个错误,或者你的程序在正常关闭服务器端socket后还是有这个问题。正如下面的这段简单的socket程序。

server.c

#include <sys/types.h>  
#include <sys/socket.h>  
#include <stdio.h>  
#include <netinet/in.h>  
#include <arpa/inet.h> 
#include <unistd.h> 
#include <stdlib.h> 
 
#define BUFFER_SIZE 40 
 
int main()   
{         
 char buf[BUFFER_SIZE];  
 int server_sockfd, client_sockfd;   
 int sin_size=sizeof(struct sockaddr_in);   
 struct sockaddr_in server_address;   
 struct sockaddr_in client_address;   
    memset(&server_address,0,sizeof(server_address));  
    server_address.sin_family = AF_INET;   
    server_address.sin_addr.s_addr = INADDR_ANY;   
    server_address.sin_port = htons(12000);   
 // 建立服务器端socket  
 if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)  
    {  
        perror("server_sockfd creation failed");  
        exit(EXIT_FAILURE);  
    }  
 // 将套接字绑定到服务器的网络地址上  
 if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)  
    {  
        perror("server socket bind failed");  
        exit(EXIT_FAILURE);  
    }  
 // 建立监听队列 
    listen(server_sockfd,5);  
 // 等待客户端连接请求到达 
    client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);  
 if(client_sockfd<0)  
    {  
        perror("accept client socket failed");  
        exit(EXIT_FAILURE);  
    }  
 // 接收客户端数据 
 if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)  
    {  
        perror("recv client data failed");  
        exit(EXIT_FAILURE);  
    }  
    printf("receive from client:%s/n",buf);  
 // 发送数据到客户端 
 if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)  
    {  
        perror("send failed");  
        exit(EXIT_FAILURE);  
    }  
    close(client_sockfd);  
    close(server_sockfd);  
    exit(EXIT_SUCCESS);  
}  

client.c

#include <sys/types.h>  
#include <sys/socket.h>  
#include <stdio.h>  
#include <netinet/in.h>                                                 
#include <arpa/inet.h>  
#include <unistd.h>  
#include <stdlib.h> 
 
#define BUFFER_SIZE 40 
 
int main()   
{   
 char buf[BUFFER_SIZE];  
 int client_sockfd;   
 int len;   
 struct sockaddr_in address;// 服务器端网络地址结构体                                            
 int result;   
    client_sockfd = socket(AF_INET, SOCK_STREAM, 0);// 建立客户端socket                                
    address.sin_family = AF_INET;   
    address.sin_addr.s_addr = inet_addr("127.0.0.1");               
    address.sin_port = htons(12000);   
    len = sizeof(address);  
 // 与远程服务器建立连接 
    result = connect(client_sockfd, (struct sockaddr *)&address, len);   
 if(result<0)   
    {   
         perror("connect failed");   
         exit(EXIT_FAILURE);   
    }   
    printf("Please input the message:");  
    scanf("%s",buf);  
    send(client_sockfd,buf,BUFFER_SIZE,0);  
    recv(client_sockfd,buf,BUFFER_SIZE,0);  
    printf("receive data from server: %s/n",buf);  
    close(client_sockfd);   
 return 0;   
}  

在成功的运行了第一次之后,当你再次启动服务器端程序时,./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

#include <sys/types.h>  
#include <sys/socket.h>  
#include <stdio.h>  
#include <netinet/in.h>  
#include <arpa/inet.h> 
#include <unistd.h> 
#include <stdlib.h> 
 
#define BUFFER_SIZE 40 
 
int main()   
{         
 char buf[BUFFER_SIZE];  
 int server_sockfd, client_sockfd;   
 int sin_size=sizeof(struct sockaddr_in);   
 struct sockaddr_in server_address;   
 struct sockaddr_in client_address;   
    memset(&server_address,0,sizeof(server_address));  
    server_address.sin_family = AF_INET;   
    server_address.sin_addr.s_addr = INADDR_ANY;   
    server_address.sin_port = htons(12000);   
 // 建立服务器端socket  
 if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)  
    {  
        perror("server_sockfd creation failed");  
        exit(EXIT_FAILURE);  
    }  
 // 设置套接字选项避免地址使用错误 
 int on=1;  
 if((setsockopt(server_sockfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on)))<0)  
    {  
        perror("setsockopt failed");  
        exit(EXIT_FAILURE);  
    }  
 // 将套接字绑定到服务器的网络地址上  
 if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)  
    {  
        perror("server socket bind failed");  
        exit(EXIT_FAILURE);  
    }  
 // 建立监听队列 
    listen(server_sockfd,5);  
 // 等待客户端连接请求到达 
    client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);  
 if(client_sockfd<0)  
    {  
        perror("accept client socket failed");  
        exit(EXIT_FAILURE);  
    }  
 // 接收客户端数据 
 if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)  
    {  
        perror("recv client data failed");  
        exit(EXIT_FAILURE);  
    }  
    printf("receive from client:%s/n",buf);  
 // 发送数据到客户端 
 if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)  
    {  
        perror("send failed");  
        exit(EXIT_FAILURE);  
    }  
    close(client_sockfd);  
    close(server_sockfd);  
    exit(EXIT_SUCCESS);  
}  

这次,让我们再次反复的启动服务器,尽情的在“黑窗户”里面输入./server ./server ./server ......服务器的程序好像突然间变乖了,呵呵,童鞋们,为自己的成就庆祝吧!!!

本文分享自微信公众号 - Golang语言社区(Golangweb)

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2016-10-06

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 微服务架构:敏捷软件架构的实际体现

    正如敏捷开发能够解决工程技术瓶颈,微服务则能够解决架构层面的瓶颈。 2014年出现的“微服务”理念仿佛一道闪电,让技术人员意识到这一全新架构风格的重要意义。面向...

    李海彬
  • Golang语言-- text/templete模板

    抽空把go 的text/templete模板整理学习了,现在总结下。 步骤 1、定义数据类型 ? ? ? ? ?

    李海彬
  • Dbdot – Golang tool to help generate Postgres schema diagrams

    dbdot is a command line tool that generates DOT description from postgres databa...

    李海彬
  • 谣传Facebook购买Coinbase,币圈大牛市即将到来?

    伴随比特币,以太坊以及各种各样的小币,币圈和区块链这两年来一直起起伏伏,吸引了无数的眼球。如今传来的最大的信息无疑是Facebook可能要买Coinbase。

    用户1564362
  • JDK 1.8 的 HashMap 详解: 为什么并发会出问题?甚至出现死循环导致系统不可用?

    为什么说HashMap是非线程安全的呢?因为在高并发情况下,HashMap在一些操作上会存在问题,如死循环问题,导致CPU使用率较高。

    一个会写诗的程序员
  • Redis主从复制

    Redis的主从复制可以把数据复制多个副本部署到其他机器,从而避免了系统中的单点问题以及满足故障恢复和负载均衡等需求。

    Java学习录
  • 光谱链—平行互联网价值传输协议及去中心化应用平台

    光谱链(即Spectrum)是基于SmartMesh基础协议致力于以去中心化的Mesh网络的形式实现万物互联的区块链底层公链。

    rectinajh
  • 什么是一致性Hash算法?

    可以将传入的 Key 按照 index=hash(key)%N 这样来计算出需要存放的节点。其中 hash 函数是一个将字符串转换为正整数的哈希映射方法,N 就...

    Java3y
  • 雪上加霜!继数据泄露后,Facebook又承认与设备制造商共享数据 | 热点

    镁客网
  • 【算法专栏】二叉树的下一个节点

    给定一个二叉树和其中的一个结点,请找出中序遍历顺序的下一个结点并且返回。注意,树中的结点不仅包含左右子结点,同时包含指向父结点的指针。

    ConardLi

扫码关注云+社区

领取腾讯云代金券