SO_REUSEADDR 套接字选项应用
在网络上的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是不会成功的。或者仅仅以为绑定的是同一个地址。就仅仅第一次启用一次。也是不行的。即使是第一次获得的套接字不须要重用,也要启用该选项,这样后面每次启用才会达到预期效果。
实例:
客户
- #include <iostream>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <stdlib.h>
- #include <strings.h>
- #include <stdio.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
- #include <errno.h>
- #include <unistd.h>
- using namespace std;
- int main()
- {
- int skfd;
- if ((skfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
- perror("socket1 error");
- exit(-1);
- }
- //启用SO_REUSEADDR
- int optval = 1;
- if (setsockopt(skfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) < 0) {
- perror("setsockopt1 error");
- exit(-1);
- }
- struct sockaddr_in saddr;
- bzero(&saddr, sizeof(saddr));
- saddr.sin_family = AF_INET;
- saddr.sin_port = htons(7777);
- saddr.sin_addr.s_addr = inet_addr("192.168.1.107");
- if (bind(skfd, (struct sockaddr*)&saddr, sizeof(saddr)) < 0) {
- perror("bind1 error");
- exit(-1);
- }
- struct sockaddr_in daddr;
- bzero(&daddr, sizeof(daddr));
- daddr.sin_family = AF_INET;
- daddr.sin_port = htons(9999);
- daddr.sin_addr.s_addr = inet_addr("192.168.1.107");
- if (connect(skfd, (struct sockaddr*)&daddr, sizeof(daddr)) < 0) {
- perror("connect1 error");
- exit(-1);
- }
- cout << "connect1 succed" << endl;
- //运行主动关闭
- close(skfd);
- sleep(10);
- if ((skfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
- perror("2 socket");
- exit(-1);
- }
- //关闭后想要重用,此时第二次启用SO_REUSEADDR,bind才会成功
- if (setsockopt(skfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) < 0) {
- perror("2 setsockopt error");
- exit(-1);
- }
- if (bind(skfd, (struct sockaddr*)&saddr, sizeof(saddr)) < 0) {
- perror("2 bind");
- exit(-1);
- }
- cout << "第二次bind成功" << endl;
- return 0;
- }
server:
- #include <iostream>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <stdlib.h>
- #include <strings.h>
- #include <stdio.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
- #include <errno.h>
- #include <unistd.h>
- #include <sys/select.h>
- #include <netinet/tcp.h>
- using namespace std;
- #define LISTENNUM 5
- int main()
- {
- int skfd;
- if ((skfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
- perror("");
- exit(-1);
- }
- struct sockaddr_in saddr;
- bzero(&saddr, sizeof(saddr));
- saddr.sin_family = AF_INET;
- saddr.sin_port = htons(9999);
- saddr.sin_addr.s_addr = htonl(INADDR_ANY);
- if (bind(skfd, (struct sockaddr*)&saddr, sizeof(saddr)) < 0) {
- perror("");
- exit(-1);
- }
- if (listen(skfd, LISTENNUM) < 0) {
- perror("");
- exit(-1);
- }
- int clifd;
- clifd = accept(skfd, NULL, NULL);
- if (clifd < 0) {
- perror("accept error");
- exit(-1);
- }
- cout << "有新连接" << endl;
- sleep(5);
- int tmp;
- while (1) {
- //read返回,server进入CLOSE_WAIT状态
- if (read(clifd, &tmp, sizeof(tmp)) == 0)
- break;
- }
- //close返回server进入LAST_ACK状态
- close(skfd);
- return 0;
- }
版权声明:本文博客原创文章,博客,未经同意,不得转载。
SO_REUSEADDR 套接字选项应用的更多相关文章
- UNIX网络编程——套接字选项(SO_REUSEADDR)
1.一般来说,一个端口释放后会等待两分钟之后才能再被使用,SO_REUSEADDR是让端口释放后立即就可以被再次使用. SO_REUSEADDR用于对TCP套接字处于TIME_WAIT状态下的sock ...
- 套接字选项 之 SO_REUSEADDR && SO_REUSEPORT
说明 本文下面内容基本上是截取自stackoverflow,针对这两个选项,在另外一篇文章中做了总结,请移步<Linux TCP套接字选项 之 SO_REUSEADDR && S ...
- UNIX网络编程——通用套接字选项
1. SO_BROADCAST 套接字选项 本选项开启或禁止进程发送广播消息的能力.只有数据报套接字支持广播,并且还必须是在支持广播消息的网络上(例如以太网,令牌环网等).我们不可能在点对点链路上进行 ...
- 通用套接字选项和TCP套接字选项
1. 套接字选项函数原型: #include <sys/socket.h> int getsockopt(int sockfd, int level, int optname, void ...
- 网络IPC:套接字之套接字选项
套接字机制提供两个套接字选项接口来控制套接字的行为.一个接口用来设置选项,另一个接口允许查询一个选项的状态.可以获取或设置的三种选项: (1)通用选项,工作在所有套接字类型上. (2)在套接字层次管理 ...
- UNIX网络编程——经常使用的套接字选项
1.设置/获取套接字选项 int setsockopt(int socket, int level, int option_name, const void *option_value, sockle ...
- UNIX网络编程——套接字选项(SOL_SOCKET级别)
#include <sys/socket.h> int setsockopt( int socket, int level, int option_name,const void *opt ...
- UNIX网络编程——套接字选项(心跳检测、绑定地址复用)
/* 设置套接字选项周期性消息检测连通性 心跳包. 心博.主要用于长连接. * 参数:套接字, 1或0开启, 首次间隔时间, 两次间隔时间, 断开次数 */ void setKeepAlive( in ...
- TCP回射客户服务器模型(02 设置套接字选项、处理多并发)
int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen); //设置套接字选项 ...
随机推荐
- C#实现环形队列
概述 看了一个数据结构的教程,是用C++写的,可自己C#还是一个菜鸟,更别说C++了,但还是大胆尝试用C#将其中的环形队列的实现写出来,先上代码: public class MyQueue<T& ...
- Lucene.Net 2.3.1开发介绍 —— 二、分词(二)
原文:Lucene.Net 2.3.1开发介绍 -- 二.分词(二) 1.2.分词的过程 1.2.1.分词器工作的过程 内置的分词器效果都不好,那怎么办?只能自己写了!在写之前当然是要先看看内置的分词 ...
- jquery ajax验证用户名是否存在(后台spring mvc)
controller层 @ResponseBody @RequestMapping(value = "/user/isExist", produces = "applic ...
- 推荐一款功能强大的js 在线编辑器
http://jszi.cn/public/oherub/11/edit
- CURD演示 2
<?php class UserAction extends Action{ public function index(){ echo "你好!"; $m=M('user' ...
- MOCKITO 应用示例
package com.paic.wms.service.auditflow.impl; import static org.junit.Assert.*; import java.util.Arra ...
- MPICH3环境配置
最新版的mpich简化了运行方式,不再提供mpd开头的命令,只需要一个mpiexec即可启动mpi运行环境,方便了mpi编程.源代码下载地址:http://www.mpich.org/download ...
- 配置虚拟主机并更改Apache默认解析路径
配置虚拟主机,非常easy 改动以下文件: 加入以下几句话 <VirtualHost *:80> ##ServerAdmin webmaster@dummy-host2.example.c ...
- [.NET Framework学习笔记]一些概念
CIL:Common Intermediate Language 公共中间语言 VB.NET 和 C#.NET 编译以后都生成相同的中间语言,程序集就是由CIL组成的,CIL代码也叫做托管代码,因为C ...
- CentOS 6.5安装Erlang/OTP 17.0
CentOS 6.5安装Erlang/OTP 17.0 作者:chszs,转载需注明.博客主页:http://blog.csdn.net/chszs Erlang眼下已经是Fedora和Debian/ ...