在网络上的SO_REUSEADDR套接字选项是用来解决地址问题重用了大量的信息。但仅仅停留在文字的表达。并没有实例,非常easy误导谁刚开始学习,和不解,此处不再赘述。

的使用该选项,以及须要注意的问题。

关于TCP断开连接四次握手,如图所看到的

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvYXNwbmV0X2x5Yw==/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="">

应用场景:

之前项目中遇到一个问题,聊天server的开启,关闭和重新启动。比如将server关闭后,实际上关闭了server的监听套接字(close),假设此时用户点击开启server,那么用户希望的情况是server又马上启动了。

而因为close后,运行了主动关闭,运行主动关闭的一端,在客户运行被动关闭之后会经历TIME_WAIT状态。如上图。TIME_WAIT的时间为1-4分钟不等。当某port处于TIME_WAIT状态时。是无法被绑定的(bind).假设在项目中用户关闭server后,希望马上启动,而程序却要等1-4分钟不等的时间后才干又一次启动server明显不是我们所希望的。SO_REUSEADDR套接字选项就能够用来解决问题。实现瞬间重新启动server

注意的地方:

1.运行主动关闭的一端进入TIME_WAIT状态,可是是有条件的,也就是对端也须要运行被动关闭。如上图。

2.SO_REUSEADDR须要在每次socket和bind前调用,“每次”也就是说,不能再想要重用的那一次。对通过socket新获取的套接字启用,比如socket->bind->close进入TIME_WAIT,而此时想要在TIME_WAIT状态下绑定,那么接下来sock->setsockopt->bind。bind是不会成功的。或者仅仅以为绑定的是同一个地址。就仅仅第一次启用一次。也是不行的。即使是第一次获得的套接字不须要重用,也要启用该选项,这样后面每次启用才会达到预期效果。

实例:

客户

  1. #include <iostream>
  2. #include <sys/types.h>
  3. #include <sys/socket.h>
  4. #include <stdlib.h>
  5. #include <strings.h>
  6. #include <stdio.h>
  7. #include <netinet/in.h>
  8. #include <arpa/inet.h>
  9. #include <errno.h>
  10. #include <unistd.h>
  11. using namespace std;
  12.  
  13. int main()
  14. {
  15. int skfd;
  16. if ((skfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
  17. perror("socket1 error");
  18. exit(-1);
  19. }
  20.  
  21. //启用SO_REUSEADDR
  22. int optval = 1;
  23. if (setsockopt(skfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) < 0) {
  24. perror("setsockopt1 error");
  25. exit(-1);
  26. }
  27.  
  28. struct sockaddr_in saddr;
  29. bzero(&saddr, sizeof(saddr));
  30. saddr.sin_family = AF_INET;
  31. saddr.sin_port = htons(7777);
  32. saddr.sin_addr.s_addr = inet_addr("192.168.1.107");
  33.  
  34. if (bind(skfd, (struct sockaddr*)&saddr, sizeof(saddr)) < 0) {
  35. perror("bind1 error");
  36. exit(-1);
  37. }
  38.  
  39. struct sockaddr_in daddr;
  40. bzero(&daddr, sizeof(daddr));
  41. daddr.sin_family = AF_INET;
  42. daddr.sin_port = htons(9999);
  43. daddr.sin_addr.s_addr = inet_addr("192.168.1.107");
  44.  
  45. if (connect(skfd, (struct sockaddr*)&daddr, sizeof(daddr)) < 0) {
  46. perror("connect1 error");
  47. exit(-1);
  48. }
  49. cout << "connect1 succed" << endl;
  50.  
  51. //运行主动关闭
  52. close(skfd);
  53. sleep(10);
  54.  
  55. if ((skfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
  56. perror("2 socket");
  57. exit(-1);
  58. }
  59.  
  60. //关闭后想要重用,此时第二次启用SO_REUSEADDR,bind才会成功
  61. if (setsockopt(skfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) < 0) {
  62. perror("2 setsockopt error");
  63. exit(-1);
  64. }
  65.  
  66. if (bind(skfd, (struct sockaddr*)&saddr, sizeof(saddr)) < 0) {
  67. perror("2 bind");
  68. exit(-1);
  69. }
  70. cout << "第二次bind成功" << endl;
  71.  
  72. return 0;
  73. }

server:

  1. #include <iostream>
  2. #include <sys/types.h>
  3. #include <sys/socket.h>
  4. #include <stdlib.h>
  5. #include <strings.h>
  6. #include <stdio.h>
  7. #include <netinet/in.h>
  8. #include <arpa/inet.h>
  9. #include <errno.h>
  10. #include <unistd.h>
  11. #include <sys/select.h>
  12. #include <netinet/tcp.h>
  13. using namespace std;
  14.  
  15. #define LISTENNUM 5
  16.  
  17. int main()
  18. {
  19. int skfd;
  20. if ((skfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
  21. perror("");
  22. exit(-1);
  23. }
  24.  
  25. struct sockaddr_in saddr;
  26. bzero(&saddr, sizeof(saddr));
  27. saddr.sin_family = AF_INET;
  28. saddr.sin_port = htons(9999);
  29. saddr.sin_addr.s_addr = htonl(INADDR_ANY);
  30.  
  31. if (bind(skfd, (struct sockaddr*)&saddr, sizeof(saddr)) < 0) {
  32. perror("");
  33. exit(-1);
  34. }
  35.  
  36. if (listen(skfd, LISTENNUM) < 0) {
  37. perror("");
  38. exit(-1);
  39. }
  40.  
  41. int clifd;
  42. clifd = accept(skfd, NULL, NULL);
  43. if (clifd < 0) {
  44. perror("accept error");
  45. exit(-1);
  46. }
  47. cout << "有新连接" << endl;
  48. sleep(5);
  49. int tmp;
  50. while (1) {
  51. //read返回,server进入CLOSE_WAIT状态
  52. if (read(clifd, &tmp, sizeof(tmp)) == 0)
  53. break;
  54. }
  55. //close返回server进入LAST_ACK状态
  56. close(skfd);
  57. return 0;
  58. }

版权声明:本文博客原创文章,博客,未经同意,不得转载。

SO_REUSEADDR 套接字选项应用的更多相关文章

  1. UNIX网络编程——套接字选项(SO_REUSEADDR)

    1.一般来说,一个端口释放后会等待两分钟之后才能再被使用,SO_REUSEADDR是让端口释放后立即就可以被再次使用. SO_REUSEADDR用于对TCP套接字处于TIME_WAIT状态下的sock ...

  2. 套接字选项 之 SO_REUSEADDR && SO_REUSEPORT

    说明 本文下面内容基本上是截取自stackoverflow,针对这两个选项,在另外一篇文章中做了总结,请移步<Linux TCP套接字选项 之 SO_REUSEADDR && S ...

  3. UNIX网络编程——通用套接字选项

    1. SO_BROADCAST 套接字选项 本选项开启或禁止进程发送广播消息的能力.只有数据报套接字支持广播,并且还必须是在支持广播消息的网络上(例如以太网,令牌环网等).我们不可能在点对点链路上进行 ...

  4. 通用套接字选项和TCP套接字选项

    1. 套接字选项函数原型: #include <sys/socket.h> int getsockopt(int sockfd, int level, int optname, void ...

  5. 网络IPC:套接字之套接字选项

    套接字机制提供两个套接字选项接口来控制套接字的行为.一个接口用来设置选项,另一个接口允许查询一个选项的状态.可以获取或设置的三种选项: (1)通用选项,工作在所有套接字类型上. (2)在套接字层次管理 ...

  6. UNIX网络编程——经常使用的套接字选项

    1.设置/获取套接字选项 int setsockopt(int socket, int level, int option_name, const void *option_value, sockle ...

  7. UNIX网络编程——套接字选项(SOL_SOCKET级别)

    #include <sys/socket.h> int setsockopt( int socket, int level, int option_name,const void *opt ...

  8. UNIX网络编程——套接字选项(心跳检测、绑定地址复用)

    /* 设置套接字选项周期性消息检测连通性 心跳包. 心博.主要用于长连接. * 参数:套接字, 1或0开启, 首次间隔时间, 两次间隔时间, 断开次数 */ void setKeepAlive( in ...

  9. TCP回射客户服务器模型(02 设置套接字选项、处理多并发)

    int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen);  //设置套接字选项 ...

随机推荐

  1. C#实现环形队列

    概述 看了一个数据结构的教程,是用C++写的,可自己C#还是一个菜鸟,更别说C++了,但还是大胆尝试用C#将其中的环形队列的实现写出来,先上代码: public class MyQueue<T& ...

  2. Lucene.Net 2.3.1开发介绍 —— 二、分词(二)

    原文:Lucene.Net 2.3.1开发介绍 -- 二.分词(二) 1.2.分词的过程 1.2.1.分词器工作的过程 内置的分词器效果都不好,那怎么办?只能自己写了!在写之前当然是要先看看内置的分词 ...

  3. jquery ajax验证用户名是否存在(后台spring mvc)

    controller层 @ResponseBody @RequestMapping(value = "/user/isExist", produces = "applic ...

  4. 推荐一款功能强大的js 在线编辑器

    http://jszi.cn/public/oherub/11/edit

  5. CURD演示 2

    <?php class UserAction extends Action{ public function index(){ echo "你好!"; $m=M('user' ...

  6. MOCKITO 应用示例

    package com.paic.wms.service.auditflow.impl; import static org.junit.Assert.*; import java.util.Arra ...

  7. MPICH3环境配置

    最新版的mpich简化了运行方式,不再提供mpd开头的命令,只需要一个mpiexec即可启动mpi运行环境,方便了mpi编程.源代码下载地址:http://www.mpich.org/download ...

  8. 配置虚拟主机并更改Apache默认解析路径

    配置虚拟主机,非常easy 改动以下文件: 加入以下几句话 <VirtualHost *:80> ##ServerAdmin webmaster@dummy-host2.example.c ...

  9. [.NET Framework学习笔记]一些概念

    CIL:Common Intermediate Language 公共中间语言 VB.NET 和 C#.NET 编译以后都生成相同的中间语言,程序集就是由CIL组成的,CIL代码也叫做托管代码,因为C ...

  10. CentOS 6.5安装Erlang/OTP 17.0

    CentOS 6.5安装Erlang/OTP 17.0 作者:chszs,转载需注明.博客主页:http://blog.csdn.net/chszs Erlang眼下已经是Fedora和Debian/ ...