当客户端保持着与服务器端的连接,这时服务器端断开,再开启服务器时会出现: 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 plaincopy
  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. #define BUFFER_SIZE 40
  9. int main()
  10. {
  11. char buf[BUFFER_SIZE];
  12. int server_sockfd, client_sockfd;
  13. int sin_size=sizeof(struct sockaddr_in);
  14. struct sockaddr_in server_address;
  15. struct sockaddr_in client_address;
  16. memset(&server_address,0,sizeof(server_address));
  17. server_address.sin_family = AF_INET;
  18. server_address.sin_addr.s_addr = INADDR_ANY;
  19. server_address.sin_port = htons(12000);
  20. // 建立服务器端socket
  21. if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)
  22. {
  23. perror("server_sockfd creation failed");
  24. exit(EXIT_FAILURE);
  25. }
  26. // 将套接字绑定到服务器的网络地址上
  27. if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)
  28. {
  29. perror("server socket bind failed");
  30. exit(EXIT_FAILURE);
  31. }
  32. // 建立监听队列
  33. listen(server_sockfd,5);
  34. // 等待客户端连接请求到达
  35. client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);
  36. if(client_sockfd<0)
  37. {
  38. perror("accept client socket failed");
  39. exit(EXIT_FAILURE);
  40. }
  41. // 接收客户端数据
  42. if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)
  43. {
  44. perror("recv client data failed");
  45. exit(EXIT_FAILURE);
  46. }
  47. printf("receive from client:%s/n",buf);
  48. // 发送数据到客户端
  49. if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)
  50. {
  51. perror("send failed");
  52. exit(EXIT_FAILURE);
  53. }
  54. close(client_sockfd);
  55. close(server_sockfd);
  56. exit(EXIT_SUCCESS);
  57. }
[c-sharp] view plaincopy
  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. #define BUFFER_SIZE 40
  9. int main()
  10. {
  11. char buf[BUFFER_SIZE];
  12. int server_sockfd, client_sockfd;
  13. int sin_size=sizeof(struct sockaddr_in);
  14. struct sockaddr_in server_address;
  15. struct sockaddr_in client_address;
  16. memset(&server_address,0,sizeof(server_address));
  17. server_address.sin_family = AF_INET;
  18. server_address.sin_addr.s_addr = INADDR_ANY;
  19. server_address.sin_port = htons(12000);
  20. // 建立服务器端socket
  21. if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)
  22. {
  23. perror("server_sockfd creation failed");
  24. exit(EXIT_FAILURE);
  25. }
  26. // 将套接字绑定到服务器的网络地址上
  27. if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)
  28. {
  29. perror("server socket bind failed");
  30. exit(EXIT_FAILURE);
  31. }
  32. // 建立监听队列
  33. listen(server_sockfd,5);
  34. // 等待客户端连接请求到达
  35. client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);
  36. if(client_sockfd<0)
  37. {
  38. perror("accept client socket failed");
  39. exit(EXIT_FAILURE);
  40. }
  41. // 接收客户端数据
  42. if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)
  43. {
  44. perror("recv client data failed");
  45. exit(EXIT_FAILURE);
  46. }
  47. printf("receive from client:%s/n",buf);
  48. // 发送数据到客户端
  49. if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)
  50. {
  51. perror("send failed");
  52. exit(EXIT_FAILURE);
  53. }
  54. close(client_sockfd);
  55. close(server_sockfd);
  56. exit(EXIT_SUCCESS);
  57. }

client.c

[c-sharp] view plaincopy
  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. #define BUFFER_SIZE 40
  9. int main()
  10. {
  11. char buf[BUFFER_SIZE];
  12. int client_sockfd;
  13. int len;
  14. struct sockaddr_in address;// 服务器端网络地址结构体
  15. int result;
  16. client_sockfd = socket(AF_INET, SOCK_STREAM, 0);// 建立客户端socket
  17. address.sin_family = AF_INET;
  18. address.sin_addr.s_addr = inet_addr("127.0.0.1");
  19. address.sin_port = htons(12000);
  20. len = sizeof(address);
  21. // 与远程服务器建立连接
  22. result = connect(client_sockfd, (struct sockaddr *)&address, len);
  23. if(result<0)
  24. {
  25. perror("connect failed");
  26. exit(EXIT_FAILURE);
  27. }
  28. printf("Please input the message:");
  29. scanf("%s",buf);
  30. send(client_sockfd,buf,BUFFER_SIZE,0);
  31. recv(client_sockfd,buf,BUFFER_SIZE,0);
  32. printf("receive data from server: %s/n",buf);
  33. close(client_sockfd);
  34. return 0;
  35. }
[c-sharp] view plaincopy
  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. #define BUFFER_SIZE 40
  9. int main()
  10. {
  11. char buf[BUFFER_SIZE];
  12. int client_sockfd;
  13. int len;
  14. struct sockaddr_in address;// 服务器端网络地址结构体
  15. int result;
  16. client_sockfd = socket(AF_INET, SOCK_STREAM, 0);// 建立客户端socket
  17. address.sin_family = AF_INET;
  18. address.sin_addr.s_addr = inet_addr("127.0.0.1");
  19. address.sin_port = htons(12000);
  20. len = sizeof(address);
  21. // 与远程服务器建立连接
  22. result = connect(client_sockfd, (struct sockaddr *)&address, len);
  23. if(result<0)
  24. {
  25. perror("connect failed");
  26. exit(EXIT_FAILURE);
  27. }
  28. printf("Please input the message:");
  29. scanf("%s",buf);
  30. send(client_sockfd,buf,BUFFER_SIZE,0);
  31. recv(client_sockfd,buf,BUFFER_SIZE,0);
  32. printf("receive data from server: %s/n",buf);
  33. close(client_sockfd);
  34. return 0;
  35. }

在成功的运行了第一次之后,当你再次启动服务器端程序时,./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 plaincopy
  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. #define BUFFER_SIZE 40
  9. int main()
  10. {
  11. char buf[BUFFER_SIZE];
  12. int server_sockfd, client_sockfd;
  13. int sin_size=sizeof(struct sockaddr_in);
  14. struct sockaddr_in server_address;
  15. struct sockaddr_in client_address;
  16. memset(&server_address,0,sizeof(server_address));
  17. server_address.sin_family = AF_INET;
  18. server_address.sin_addr.s_addr = INADDR_ANY;
  19. server_address.sin_port = htons(12000);
  20. // 建立服务器端socket
  21. if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)
  22. {
  23. perror("server_sockfd creation failed");
  24. exit(EXIT_FAILURE);
  25. }
  26. // 设置套接字选项避免地址使用错误
  27. int on=1;
  28. if((setsockopt(server_sockfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on)))<0)
  29. {
  30. perror("setsockopt failed");
  31. exit(EXIT_FAILURE);
  32. }
  33. // 将套接字绑定到服务器的网络地址上
  34. if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)
  35. {
  36. perror("server socket bind failed");
  37. exit(EXIT_FAILURE);
  38. }
  39. // 建立监听队列
  40. listen(server_sockfd,5);
  41. // 等待客户端连接请求到达
  42. client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);
  43. if(client_sockfd<0)
  44. {
  45. perror("accept client socket failed");
  46. exit(EXIT_FAILURE);
  47. }
  48. // 接收客户端数据
  49. if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)
  50. {
  51. perror("recv client data failed");
  52. exit(EXIT_FAILURE);
  53. }
  54. printf("receive from client:%s/n",buf);
  55. // 发送数据到客户端
  56. if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)
  57. {
  58. perror("send failed");
  59. exit(EXIT_FAILURE);
  60. }
  61. close(client_sockfd);
  62. close(server_sockfd);
  63. exit(EXIT_SUCCESS);
  64. }
[c-sharp] view plaincopy
  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. #define BUFFER_SIZE 40
  9. int main()
  10. {
  11. char buf[BUFFER_SIZE];
  12. int server_sockfd, client_sockfd;
  13. int sin_size=sizeof(struct sockaddr_in);
  14. struct sockaddr_in server_address;
  15. struct sockaddr_in client_address;
  16. memset(&server_address,0,sizeof(server_address));
  17. server_address.sin_family = AF_INET;
  18. server_address.sin_addr.s_addr = INADDR_ANY;
  19. server_address.sin_port = htons(12000);
  20. // 建立服务器端socket
  21. if((server_sockfd = socket(AF_INET, SOCK_STREAM, 0))<0)
  22. {
  23. perror("server_sockfd creation failed");
  24. exit(EXIT_FAILURE);
  25. }
  26. // 设置套接字选项避免地址使用错误
  27. int on=1;
  28. if((setsockopt(server_sockfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on)))<0)
  29. {
  30. perror("setsockopt failed");
  31. exit(EXIT_FAILURE);
  32. }
  33. // 将套接字绑定到服务器的网络地址上
  34. if((bind(server_sockfd,(struct sockaddr *)&server_address,sizeof(struct sockaddr)))<0)
  35. {
  36. perror("server socket bind failed");
  37. exit(EXIT_FAILURE);
  38. }
  39. // 建立监听队列
  40. listen(server_sockfd,5);
  41. // 等待客户端连接请求到达
  42. client_sockfd=accept(server_sockfd,(struct sockaddr *)&client_address,(socklen_t*)&sin_size);
  43. if(client_sockfd<0)
  44. {
  45. perror("accept client socket failed");
  46. exit(EXIT_FAILURE);
  47. }
  48. // 接收客户端数据
  49. if(recv(client_sockfd,buf,BUFFER_SIZE,0)<0)
  50. {
  51. perror("recv client data failed");
  52. exit(EXIT_FAILURE);
  53. }
  54. printf("receive from client:%s/n",buf);
  55. // 发送数据到客户端
  56. if(send(client_sockfd,"I have received your message.",BUFFER_SIZE,0)<0)
  57. {
  58. perror("send failed");
  59. exit(EXIT_FAILURE);
  60. }
  61. close(client_sockfd);
  62. close(server_sockfd);
  63. exit(EXIT_SUCCESS);
  64. }

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

Address already in use的解决方法相关推荐

  1. docker出现Error starting userland proxy: listen tcp4 0.0.0.0:3306: bind: address already in use的解决方法

    目录 1. 问题所示 2. 分析原理 3. 解决方法 1. 问题所示 出现如下问题 (base) gaokaoli@kkjz-ThinkStation-P330:~$ sudo docker star ...

  2. Linux下Socket编程:bind().Address already in use的解决方法

    当客户端保持着与服务器端的连接,这时服务器端断开,再开启服务器时会出现: Address already in use,可以用netstat -anp | more 可以看到客户端还保持着与服务器的连 ...

  3. anywhere执行时端口被占用Address already in use:8080解决方法

    方法如下,在cmd命令模式下输入netstat -ano 然后找到占用8080端口的那个进程! 如图所示: 我们可以看到占用8080端口进程的PID号是11672,然后我们要做的就是去找到这个进程, ...

  4. linux qemu 报错 Unable to reserve 0xfffff000 bytes of virtual address space at 0x1000 解决方法

    目录 错误现象 可能原因 解决过程 解决方法 错误现象 使用 qemu-i686 运行 i686平台文件 报错 qemu-i686: Unable to reserve 0xfffff000 byte ...

  5. 对于高并发短连接造成Cannot assign requested address解决方法

    对于高并发短连接造成Cannot assign requested address解决方法 参考文章: (1)对于高并发短连接造成Cannot assign requested address解决方法 ...

  6. java.net.BindException: Address already in use: 解决方法

    java.net.BindException: Address already in use: 解决方法 参考文章: (1)java.net.BindException: Address alread ...

  7. docker 报错 non-overlapping IPv4 address pool among the defaults to assign to the network 解决方法

    目录 错误现象 错误原因 解决方法 方法一 删除没使用的网络 方法二 指定网络配置 方法三 修改docker默认网络地址(推荐) 错误现象 docker-compose up 报错 Error res ...

  8. Check failed: error == cudaSuccess (74 vs. 0) misaligned address的解决方法

    今天训练网络的时候加入了Deconvolution Layer,结果一运行报Check failed: error == cudaSuccess (74 vs. 0) misaligned addre ...

  9. hadoop 集群启动时 Address already in use 解决方法

    搭建好hadoop集群之后,运行start-all.sh脚本,发现有如下报错 starting namenode, logging to /usr/local/hadoop/libexec/../lo ...

最新文章

  1. UVA 10306 e-Coins(全然背包: 二维限制条件)
  2. UITableView reloadData 跳动闪动
  3. 注意:宝塔曝未授权访问数据库漏洞,尽快升级版本!
  4. 飞鸽传书也在2010年免费发布了
  5. c语言用if如何删除末尾空格,新人提问:如何将输出时每行最后一个空格删除
  6. OpenCV4每日一练day3:运行OpenCV示例程序(物体跟踪)
  7. Pychram中如何导入其他文件的.py文件
  8. 《Tomcat权威指南》读书笔记
  9. 关系数据库第一第二第三范式
  10. 密码行业标准培训小结
  11. 电商平台--Mysql主从搭建(2)
  12. android小说阅读、MVP + RxJava + Retrofit项目、证件拍照裁剪、蓝牙锁等源码器
  13. Apache Flink 1.14.4 on yarn ha环境搭建
  14. python正则匹配txt特定字符串(有换行)
  15. VBA编程_Application
  16. vue前端项目启动报错,node-sass安装失败
  17. delegate类型
  18. 油猴Tampermonkey简介
  19. 在虚拟和现实里游刃有余:增强现实技术
  20. 不定积分-分子分母次数较高

热门文章

  1. 纠偏控制比例阀放大器
  2. MMDetection系列之(迁移学习)
  3. Facebook将其未来押在元宇宙上——但它到底是什么?
  4. lte协议软件 开源的4g_4G LTE 协议中又现新缺陷
  5. 易康插件ESP2及其软件的详细使用
  6. vivoX7的Android版本哪个好,vivo x7有几个版本?vivo x7各版本区别对比评测
  7. C#如何获取当前控件的子控件以及多重子控件的实例(能操作这些获取到的子控件)
  8. 更改win7 64位系统调用IE时,默认64位IE的方法
  9. SQL 注入 学习笔记 第十二天
  10. 网易邮箱大师日历功能的同步问题