河北常见网站建设价格百度做网站需要多少钱

news/2025/9/30 13:57:44/文章来源:
河北常见网站建设价格,百度做网站需要多少钱,搭建安装wordpress教程,惠州seo招聘目录 一、TCP 基本信息 1.1、TCP 的头格式 1.2、什么是 TCP 1.3、什么是 TCP 连接 1.4、TCP 与 UDP 的区别 1.2、TCP 连接建立 1.2.1、TCP 三次握手的过程 1.2.2、为什么是三次握手#xff1f;不是两次#xff1f;四次#xff1f;#xff08;这个问题真是典中典不是两次四次这个问题真是典中典 1.2.2、为什么每次建立 TCP 连接时初始化的序列号都要求不一样呢 1.2.3、初始化序列号 ISN 是如何随机产生的 1.2.4、既然 IP 层会分片为什么 TCP 层还需要 MSS 呢 1.2.5、第一次握手丢失了会发生什么 1.2.6、第二次握手丢失了会发生什么 1.2.7、第三次握手丢失了会怎么办 1.2.8、什么是 SYN 攻击如何避免 SYN 攻击 1.3、TCP 连接断开 1.3.1、TCP 四次挥手是怎样的 1.3.3、第一次挥手丢失了会发生什么 1.3.4、第二次挥手丢失了会发生什么 1.3.5、第三次挥手丢失了会发生什么 1.3.6、如果第四次挥手丢失了会怎么办 1.3.7、为什么 TIME_WAIT 等待的时间是 2MSL 1.3.8、为什么需要 TIME_WAIT 状态 1.3.9、TIME_WAIT 过多有什么危害 1.3.10、服务器出现大量 TIME_WAIT 状态的原因有哪些 1.3.11、服务器出现大量 CLOSE_WAIT 状态的原因有哪些 1.3.12、如果已经建立连接但是客户端突然发生故障怎么办 1.3.13、如果已经建立连接但是服务端的进程崩溃会发生什么 1.4、Socket 编程 1.4.1、针对 TCP 应该如何 Socket 编程 1.4.2、listen 参数 backlog 的意义 1.4.3、accept 发生在三次握手的哪一步 1.1.4、客户端调用 close 了连接是断开的流程是什么 1.1.5、没有 accept 能建立 TCP 连接吗 1.1.6、没有 listen能建立 TCP 连接吗 一、TCP 基本信息 1.1、TCP 的头格式 序列号在建立连接时由计算机生成的随机数作为其初始值通过 SYN 包传给接收端主机每发送一次数据就【累加】一次该【数据字节数】的大小。用来解决网络包乱序的问题。 确认应答号指下一次【期望】收到的数据的序列号发送端收到这个确认应答以后可以认为在这个序号以前的数据都已经被正常接收。用来解决丢包的问题。 控制位 ACK 该位为 1 时【确认应答】的字段变为有效TCP 规定除了最初建立连接时的 SYN 包之外该位必须设置为 1。 RST 该位设置为 1 时表示 TCP 连接中出现异常必须强制断开连接。 SYN 该位设置为 1 时表示希望建立连接并在其【序列号】的字段进行序列号初始值的设定。 FIN 该位为 1 时表示今后不会再有数据发送希望断开连接。当通信希望断开连接时通信双方的主机之间就可以互相交换 FIN 位为 1 的 TCP 段。 1.2、什么是 TCP TCP 是面向连接的、可靠的、基于字节流的传输层通信协议。能够确保接收端接受的网络包是无损坏、无间隔、非冗余和按序的。 面向连接一定是【一对一】才能连接不能像 UDP 协议可以一个主机同时向多个主机发送消息也就是一对多是无法做到的。 可靠的无论网络链路中出现了怎样的链路变化TCP 都可以保证一个报文一定能够达到接收端。 字节流用户消息通过 TCP 协议传输时消息可能会被操作系统【分组】成多个 TCP 报文如果接收方的程序如果不知道【消息的边界】是无法读出一个有效的用户消息的。并且 TCP 报文是【有序的】当【前一个】TCP 报文没有收到的时候即使它先收到了后面的 TCP 报文那么也不能扔给应用层去处理同时对【重复】的 TCP 保报文会自动丢弃。 1.3、什么是 TCP 连接 连接用于保证可靠性和流量控制维护的某些状态信息这些信息的组合包括 Socket序列号和窗口大小称为连接。 TCP 连接通过四元组确定一个连接源地址源端口目的地址目的端口 源地址和目的地址的字段32位是在 IP 头部中作用是通过 IP 协议发送报文给对方主机。 源主机和目的端口的字段16位是在 TCP 头部中作用是告诉 TCP 协议应该把报文发送给哪个进程。 有一个 IP 的服务端监听了一个端口它的 TCP 的最大连接数是多少 最大 TCP 连接数 客户端的 IP 数 * 客户端的端口数 当然服务端最大并发 TCP 数远不能达到理论的上限会有如下影响 文件描述符限制每个 TCP 连接都是一个文件如果文件描述符被占满了会发生 Too many open files。Linux 对可打开的文件描述符的数量分别作了三个方面的限制 系统级当前系统可打开的最大数量通过 cat /proc/sys/fs/file-max 查看用户级指定用户可打开的最大数量通过 cat /etc/security/limits.conf 查看进程级单个进程可打开的最大数量通过 cat /proc/sys/fs/nr_open 查看 内存限制每个 TCP 连接都要占用一定内存操作系统的内存是有限的如果内存资源被占满后会发生 OOMOut of memory。 1.4、TCP 与 UDP 的区别 UDP 不提供复杂的控制机制利用 IP 提供面向【无连接】的通信服务。 UDP 头部格式如下 目标和源端口主要是告诉 UDP 协议应该把报文发给哪个进程包长度该字段保存了 UDP 首部的长度跟数据的长度之和校验和校验和是为了提供可靠的 UDP 首部和数据而设计防止收到在网络传输中受损的                     UDP 包 TCP 和 UDP 区别 ①、连接 TCP 是面向连接的传输层协议传输数据前先要建立连接UDP 是不需要建立连接即刻传输数据 ②、服务对象 TCP 是一对一的两点服务即一条连接只有两个端点UDP 支持一对一一对多多对多的交互通信 ③、可靠性 TCP 是可靠交付数据的数据可以无差错不丢失不重复按序到达UDP 是最大努力交付不保证可靠交付数据。但是我们可以基于 UDP 传输协议实现一个可靠的传输协议比如 QUIC 协议。 ④、拥塞控制、流量控制 TCP 有拥塞控制和流量控制机制保证数据传输的安全性UDP 则没有即使网络非常拥堵了也不会影响 UDP 的发送速率 ⑤、首部开销 TCP 首部长度较长会有一定的开销首部在没有使用【选项】字段时是 20 个字节如果使用了【选项】字段则会边长UDP 首部只有 8 个字节并且是固定不变的开销较小 ⑥、传输方式 TCP 是流式传输没有边界但保证顺序和可靠UDP 是一个包一个包的发送是有边界的但可能会丢包和乱序 ⑦、分片不同 TCP 的数据大小如果大于 MSS 大小则会在传输层进行分片目标主机收到后也同样在传输层组装 TCP 数据包如果中途丢失了一个分片只需要传输丢失的这个分片UDP 的数据大小如果大于 MTU 大小则会在 IP 层进行分片目标主机收到后会在 IP 层组装完数据接着再传给传输层 TCP 和 UDP 应用场景 TCPFTP 文件传输。HTTP / HTTPSUDP包总量较少的通信如 DNSSNMP 等。视频音频等多媒体通信。广播通信 为什么 UDP 头部没有【首部长度】字段而 TCP 头部有呢 原因是 TCP 有可变长的【选项字段】而 UDP 头部长度则是不会变化的无需多一个字段去记录 UDP 的首部长度 为什么 UDP 头部有【包长度】字段而 TCP 头部则没有呢 TCP 数据的长度 IP 总长度 - IP 首部长度 - TCP 首部长度 但是 UDP 也是基于 IP 层的呀那 UDP 也应该可以通过这个公式计算呀有两种靠谱的说法 因为为了网络设备硬件设计和处理方便首部长度需要是 4 字节的整数倍。如果去掉 UDP 的【包长度】字段那 UDP 首部长度就不是 4 字节的整数倍了可能是为了补全为 4 字节的整数倍。如今的 UDP 是基于 IP 协议发展的但是当年可能不是依赖的可能是别的不提供自身报文长度或首部长度的网络层协议因此需要长度字段以供计算。 TCP 和 UDP 可以使用同一个端口吗 可以的在数据链路层中通过 MAC 地址来寻找局域网中的主机。在网际层中通过 IP 地址来寻找网络中互联的主机或服务器。在传输层中需要通过端口来进行寻址来识别同一计算机同时通信的不同应用程序。所以传输层的【端口号】的作用是为了区分同一个主机上不同应用程序的数据包。传输层有两个传输协议分别是 TCP 和 UDP 在内核中是两个完全独立的软件模块当主机收到数据包后可以在 IP 包头的【协议号】字段知道该数据包是 TCP/UDP 所以可以根据这个信息确定发送给哪个模块处理 送给 TCP/UDP 模块的报文根据【端口号】确定送给哪个应用程序处理。因此TCP/UDP 各自的端口号也相互独立如 TCP 中有一个 80 端口UDP 中也有一个 80 端口二者不冲突。 1.2、TCP 连接建立 1.2.1、TCP 三次握手的过程 一开始客户端和服务端都处于 CLOSE 状态。先是由服务端主动监听某个端口处于 LISTEN 状态。 客户端会随机初始化序号client_isn将此序号置于 TCP 首部的【序号】字段中同时把 SYN 标志置为 1 表示 SYN 报文。接着把第一个 SYN 报文发送给服务端表示向服务端发起连接该报文不包括应用层数据之后客户端处于 SYN-SENT 状态。 服务端收到客户端的 SYN 报文后首先服务端也随机初始化自己的序号server_isn将此序号填入 TCP 首部的【序号】字段中其次把 TCP 首部的【确认应答号】字段填入       【client_isn 1】接着把 SYN 和 ACK 标志位置都置为 1 .最后把该报文发给客户端该报文也不包含应用层数据之后服务端处于 SYN-RECD 状态。 客户端收到服务端报文之后还要向服务端回应最后一个应答报文首先该应答报文 TCP 首部 ACK 标志置为 1 其次【确认应答号】字段填入 server_isn 1最后把报文发送给服务端这次报文可以携带客户到服务端的数据之后客户端处于 ESTABLISHED 状态。 服务器收到客户端的应答报文后也进入 ESTABLSHED 状态。 从上述的过程中可以发现第三次握手是可以携带数据的前两次握手是不可以携带数据的。 一旦完成三次握手双方都处于 ESTABLISHED 状态此时连接就已建立完成客户端和服务端就可以相互发送数据了。 1.2.2、为什么是三次握手不是两次四次这个问题真是典中典 在前面我们都知道 TCP 连接用于保证可靠性和流量控制维护的某些状态信息这些信息的组合包括 Socket序列号和窗口大小称为连接。 所以最重要的是为什么三次握手才可以初始化 Socket 序列号 和窗口大小并建立 TCP 连接 从下面三个方面开始分析 三次握手才可以阻止重复历史连接的初始化主要原因三次握手才可以保证同步双方的初始化序列号三次握手才可以避免资源浪费 ①、避免历史连接 三次握手的首要原因是为了防止旧的重复连接初始化造成混乱。 举个 客户端先发送了 SYN 报文seq 90报文然后客户端宕机了而且这个 SYN 报文还被网络阻塞了服务端并没有收到接着客户端重启后又重新向服务端建立连接发送了 SYNseq 100报文注意不是重传重传的 SYN 的序列号是一样的 客户端连续多次发送 SYN 都是同一个四元组建立的报文在网络拥塞情况下 一个【旧 SYN 报文】比【最新的 SYN】报文早到达了服务端那么此时服务端就会回一个 SYN ACK 报文给客户端此报文中的确认号是 9190 1客户端收到后发现自己期望收到的确认好应该是 100 1而不是 90 1于是就会回 RST 报文服务端收到 RST 报文后就会释放连接后续最新的 SYN 抵达了服务端后客户端与服务端就可以正常的完成三次握手了 上述中的【旧 SYN 报文】称为历史连接TCP 使用三次握手建立连接的最主要原因是防止【历史连接】初始化了连接。 Tip可能有疑问如果服务端在收到 RST 报文之前先收到了【新 SYN 报文】也就是服务端收到客户端报文的顺序是【旧 SYN 报文】 - 【新 SYN 报文】此时会发生什么呢 当服务端第一次收到 SYN 报文也就是收到【旧 SYN 报文】时就会回复 SYN ACK 报文给客户端此报文中的确认号是 9190 1然后此时再收到【新 SYN 报文】时就会回 Challenge Ack 报文给客户端这个 ACK 报文并不是确认收到【新 SYN 报文】的而是上一次的 ACK 确认号也就是 9190 1所以客户端收到此 ACK 报文时发现自己期望收到的确认号应该是101而不是91于是就会回 RST 报文。 如果是两次握手的话就无法阻止历史连接那为什么 TCP 两次握手无法阻止历史连接呢 是因为在两次握手的情况下服务端没有中间状态给客户端来阻止历史连接导致服务端可能建立一个历史连接造成资源浪费。 想一想在两次握手的情况下服务端在收到 SYN 报文后就进入了 ESTABLISHED 状态意味着这时可以给对方发送数据但是客户端此时还没有进入 ESTABLISHED 状态假设此次是历史连接客户端判断到此次连接为历史连接那么就会回 RST 报文来断开连接而服务端在第一次握手的时候就进入 ESTABLISHED 状态所以它可以发送数据的但是并不知道这个是历史连接他只有在收到 RST 报文后才会断开连接。 可以看到如果采用两次握手建立 TCP 连接的场景下服务端在向客户端发送数据时并没有阻止掉历史连接导致服务端建立了一个历史连接又白白发送了数据妥妥的浪费了服务器资源。 因此要解决这个问题最好就是在服务端发送数据前也就是建立连接之前要阻止掉历史连接这样就不会造成资源浪费而要实现这个就得三次握手而不是两次。 Tip客户端发送第三次握手后就可以发送数据了而被动方此时还是 syn_received 状态如果 ACK 丢了那客户端发的数据是不是也白白浪费了 不是的即使服务端还是在 syn_received 状态受到了客户端发送的数据还是可以建立连接的并且还可以正常收到这个数据包。这是因为数据报文中是有 ACK 标识位也有确认号这个确认号就是确认收到了第二次握手所以这个服务端收到了这个数据报文是可以正常建立连接的然后就可以正常接收这个数据包了。 ②、同步双方初始序列号 TCP 协议的通信双方都必须维护一个【序列号】序列号是可靠传输的一个关键因素作用 接收方可以去除重复的数据接收方可以根据数据包的序列号按需接收可以标识发送出去的数据包中哪些是已经被对方收到的通过 ACK 报文中的序列号知道 可见序列号在 TCP 连接中占据着非常重要的作用所以当客户端发送携带【初始序列号】的 SYN 报文的时候需要服务端回一个 ACK 应答报文表示客户端的 SYN 报文已经被服务端成功接收那当服务端发送【初始序列号】给客户端的时候依然也要得到客户端的应答回应这样一来一回才能确保双方的初始序列号能被可靠的同步。 四次握手其实也能够可靠的同步双方的初始化序列号但由于第二步和第三步可以优化成一步所以就成了【三次握手】。而两次握手只保证了一方的初始化序列号能够被对方正常接收。 ③、避免资源浪费 如果只有【两次握手】当客户端发生的 SYN 报文在网络中阻塞客户端没有接收到 ACK 报文就会重新发送 SYN由于没有第三次握手服务器不清楚客户端是否收到了自己回复的 ACK报文所以服务器每收到一个 SYN 就只能主动建立一个连接这会造成什么情况呢 如果客户端发送的 SYN 报文在网路中阻塞了重复发送很多次 SYN 报文那么服务端在收到请求之后就会建立多个冗余的无效链接造成不必要的资源浪费。 即两次握手会造成消息滞留情况下服务端重复接受无用的连接请求 SYN 报文而造成重复分配资源。 Tip很多人说两次握手不是也能够根据上下文信息丢弃 SYN 历史报文吗 这里两次握手是假设【由于没有第三次握手服务端不清楚客户端是否已经收到了自己发送的建立连接的 ACK 确认报文所以每收到一个 SYN 报文就只能先主动建立一个连接】这个场景。 1.2.2、为什么每次建立 TCP 连接时初始化的序列号都要求不一样呢 主要原因有两个方面 为了防止历史报文被下一个相同四元组的连接接收主要方面为了安全性防止黑客伪造的相同序列号的 TCP 报文被对方接收 ①、假设每次建立连接客户端和服务端的初始化序列号都是从 0 开始 过程如下 客户端和服务端建立一个 TCP 连接在客户端发送数据包被阻塞了然后超时重传了这个数据包而此时服务端设备断电重启了之前与客户端建立的连接消失了于是在收到客户端的数据包的时候就会发送 RST 报文。紧接着客户端又与服务端建立了与上一个连接相同四元组的连接在新连接建立完成后上一个连接中被网络阻塞的数据包正好抵达了服务端刚好该数据的序列号正好是在服务端的接收窗口内所以该数据包会被服务端正常接收就会造成数据错乱。 如果每次建立连接客户端和服务端的初始化序列号都是一样的话很容易出现历史报文被下一个相同四元组的连接接收的问题。 如果每次建立连接客户端和服务端的初始化序列号都【不一样】就有大概率因为历史报文的序列号【不在】对方接收窗口从而很大程度上避免了历史报文如下图 相反如果每次建立连接客户端与服务端的初始化序列号都一样就有大概率遇到历史报文的序列号刚好在对方的接收窗口内从而导致历史报文被重新接收。注意这只是很大程度上避免了这种情况并不是完全避免了因为序列号会有回绕的情况所以需要用时间戳的机制来判断历史报文 1.2.3、初始化序列号 ISN 是如何随机产生的 起始 ISN 是基于时钟的每 4 微妙 1转一圈要 4.55 个小时。 RFC793 提到初始化序列号 ISN 随机生成算法 ISN M Flocalhost localport remotehost remoteport。M 是一个计时器 这个计时器每隔 4 微妙加 1 。F 是一个 Hash 算法根据源IP、目的IP、源端口、目的端口生成一个随机数值要保证 Hash 算法不能轻易被外部推算出。 可以看到随机数是会基于时钟计时器递增的基本不可能会随机生成一样的初始化序列号。 1.2.4、既然 IP 层会分片为什么 TCP 层还需要 MSS 呢 我们先来认识一下 MTU 和 MSS MTU一个网络包的最大长度以太网中一般为 1500 字节MSS除去 IP 和 TCP 头部之后一个网络包所能容纳的 TCP 数据的最大长度 如果在 TCP 的整个报文头部 数据交给 IP 层进行分片会有什么异常呢 当 IP 层有一个超过 MTU 大小的数据TCP 头部 TCP 数据要发送那么 IP 层就要进行分片把数据分片成若干片保证每个分片都小于 MTU。把一份 IP 数据报进行分片后由目标主机的 IP 层来进行重新组装后再交由上一层 TCP 传输层。 这看起来很好其实存在隐患那么如果一个 IP 分片丢失整个 IP 报文的所有分片都得重传。因为 IP 层本身是没有超时重传机制它由传输层的 TCP 来负责超时和重传。 当某一个 IP 分片丢失后接收方的 IP 层就无法组装成一个完整的 TCP 报文头部 数据也就无法将数据报文送到 TCP 层所以接收方不会响应 ACK 给发送方因为发送方迟迟收不到 ACK 确认报文所以会触发超时重传就会重发【整个 TCP 报文头部 数据】。 因此可以知道由 IP 层进行分片传输是非常没有效率的。所以为了达到最佳的传输性能TCP 协议在建立连接的时候通常要协商双方的 MSS 值当 TCP 层发现数据超过 MSS 时则就会先进行分片当然由它形成的 IP 包的长度也就不会大于 MTU自然也就不会用 IP 分片啦~ 经过 TCP 层分片之后如果一个 TCP 分片丢失后进行重发时也是以 MSS 为单位而不用重传所有的分片大大增加了重传的效率。 1.2.5、第一次握手丢失了会发生什么 当客户端想和服务端建立 TCP 连接的时候首先第一个发的就是 SYN 报文然后进入到 SYN_SENT 状态。在这之后如果客户端迟迟收不到服务端的 SYN-ACK 报文第二次握手就会触发【超时重传】机制重传 SYN 报文而且重传的 SYN 报文的序列号都是一样的。 不同操作系统的超时时间不同有的 1s有的 3s 这个超时时间是写死在内核中的。 当客户端在 1s 之后没收到客户端的 SYN-ACK 报文后客户端就会重发 SYN 报文那到底重发几次呢 在 Linux 中客户端的 SYN 报文最大重传次数由 tcp_syn_retries 内核参数控制这个参数是可以自定义的默认值为 5cat /proc/sys/net/ipv4/tcp_syn_retries通常第一次超时重传是在 1s 后第二次超时重传是在 2s 第三次超时重传是在 4s 后第四次超时重传是在 8s 后第五次是在超时重传 16s 后。每次超时重传的时间是上一次的 2 倍。 当第五次超时重传后会继续等待 32s 如果客户端仍然没有回应 ACK客户端就不再发送 SYN 包然后断开 TCP 连接。所以总耗时为1248163263s 大概是一分钟左右 举个假设 tcp_syn_retries 参数值是 3那么当客户端的 SYN 报文一直在网络中丢失时会发生下个过程 具体过程 当客户端重传 3 次 SYN 报文之后由于 tcp_syn_retries 3已达到最大重传次数于是再等待一段时间如果还是没收到服务端的第二次握手那么客户端就会断开连接。 1.2.6、第二次握手丢失了会发生什么 当服务端收到客户端的第一次握手后就会回 SYN-ACK 报文给客户端这个就是第二次握手此时服务端会进入 SYN_RCVD 状态。 第二次握手的 SYN-ACK 报文其实有两个目的 第二次握手里的 ACK 是对第一次握手的确认报文第二次握手里的 SYN是服务端发起建立 TCP 连接的报文 如果第二次报文丢失了就会发生有趣的事情 因为第二次报文里是对客户端的第一次握手的 ACK 确认报文所以如果客户端迟迟没有收到第二次握手那么客户端就觉得是自己的 SYN 报文第一次握手丢失了于是客户端就会触发超时重传机制重传 SYN 报文。 然后因为第二次握手中包含服务端的 SYN 报文所以当客户端收到后需要给服务端发送 ACK 确认报文第三次握手服务器才会认为该 SYN 报文被客户端收到了。 那么如果第二次握手丢失了服务端就会收不到第二次握手但是能收到客户端重发的第一次握手但是回复的第二次握手客户端也一直收不到当客户端不再重发最后一次第一次握手时服务端发送第二次握手但还是收不到于是服务端这边会触发超时重传机制重传 SYN-ACK 报文。在 Linux 下SYN-ACK 报文的最大重传次数由 tcp_synack_retries 内核参数决定的默认值是 5cat /proc/sys/net/ipv4/tcp_synack_retries因此当第二次握手丢失了客户端和服务端都会重传 客户端会重传 SYN 报文也就是第一次握手最大重传次数由 tcp_syn_retries 内核参数决定 服务端会重传 SYN-ACK 报文也就是第二次握手最大重传次数由 tcp_synack_retries 参数决定 举个。假设 tcp_syn_retries 参数值为 1tcp_synack_retries 参数值为 2 那么当第二次握手一直丢失时发生的过程如下 具体过程 当客户端重传 1 次 SYN 报文后由于 tcp_syn_retries 为 1 以达到最大重传次数于是再等待一段时间如果还没能收到服务端的第二次握手SYN-ACK 报文那么客户端就会断开连接。 当服务端超时重传 2 次 SYN-ACK 报文后由于 tcp_synack_retries 为 2已达到最大重传次数于是再等待一段时间如果还是没能收到客户端的第三次握手那么服务端就会断开连接。 1.2.7、第三次握手丢失了会怎么办 客户端收到服务端的 SYN-ACK 报文后就会给服务端回一个 ACK 报文也就是第三次握手此时客户端状态进入到 ESTABLISH 状态。 因为这个第三次握手的 ACK 是对第二次握手的 SYN 的确认报文所以当第三次握手丢失了如果服务端那一方迟迟收不到这个确认报文就会触发超时重传机制重传 SYN-ACK 报文直到收到第三次握手或者达到最大重传次数。注意ACK 报文是不会有重传的当 ACK 丢失了就由对方重传对应的报文。 举个假设 tcp_synack_retries 参数值为 2 那么当第三次握手一直丢失时发生的过程如下 具体过程 当服务器超时重传 2 次 SYN-ACK 报文后由于 tcp_synack_retries 为 2已达到最大重传次数于是再等待一段时间如果还是没能收到客户端的第三次握手那么服务器就会断开连接。 1.2.8、什么是 SYN 攻击如何避免 SYN 攻击 我们都知道 TCP 连接建立是需要三次握手的假设攻击者短时间伪造不同 IP 地址的 SYN 报文服务端就会接收到一个 SYN 报文就进入到 SYN_RCVD 状态但服务端发送出去的 ACKSYN 报文无法得到未知 IP 主机的 ACK 应答久而久之就会沾满服务端的半连接队列使得服务端不能为正常用户服务。 正常流程 当服务器接收到客户端的 SYN 报文时会创建一个半连接的对象然后将其加入到内核的【SYN 队列】接着发送 SYN ACK 给客户端等待客户端回应 ACK 报文服务端接收到 ACK 报文后从【SYN 队列】取出一个半连接对象然后创建一个新的连接对象放入到【Accept 队列】应用通过调用 accpet() Socket 接口从【Accept 队列】取出连接对象 避免 SYN 攻击方式可以有以下四种 调大 netdev_max_backlog当网卡接收数据包的速度大于内核处理的速度时会有一个队列保存这些数据包控制该队列的最大值参数默认是 1000我们要适当调大该参数比如设置为 10000增大 TCP 半连接队列开启 tcp_syncookies减少 SYN ACK 重传次数 详细说一下 3 开启 syncookies 功能就可以在不使用 SYN 半连接队列的情况下成功建立连接相当于绕过了 SYN 半连接队列来建立连接。 具体过程 当【SYN 队列】满之后后续服务端收到 SYN 包不会丢弃而是根据算法计算出一个 cookies 值将 cookie 值放到第二次握手报文的【序列号】里然后服务端回第二次握手给客户端服务端收到客户端的应答报文时服务端回检查这个 ACK 包的合法性。如果合法将该连接对象放入到【Accept 队列】最后应用程序通过调用 accpet() 接口从【Accept 队列】取出的连接 可以看到当开启了 tcp_syncookies 了即使受到 SYN 攻击而导致 SYN 队列满时也能保证正常的连接成功建立。net.ipv4.tcp_syncookies 参数主要有三个值 0 值表示关闭该功能1 值表示仅当 SYN 半连接队列放不下时再启用它2 值表示无条件开启功能 1.3、TCP 连接断开 1.3.1、TCP 四次挥手是怎样的 TCP 断开是通过四次挥手方式执行的。双方都可以主动断开连接断开连接后主机中的【资源】将被释放四次挥手的过程如下 客户端打算关闭连接此时会发送一个 TCP 首部 FIN 标志位被置为 1 的报文也即 FIN 报文之后客户端进入 FIN_WAIT_1 状态服务端收到该报文之后就向客户端发送 ACK 应答报文接着服务器进入 CLOSE_WAIT 状态客户端收到服务端的 ACK 应答报文后之后进入 FIN_WAIT_2 状态等待服务端处理完数据后也向客户端发送 FIN 报文之后服务端进入 LAST_ACK 状态客户端收到服务端的 FIN 报文之后回一个 ACK 应答报文之后进入 TIME_WAIT 状态服务端收到了 ACK 应答报文后就进入了 CLOSE 状态至此服务端已经完成连接的关闭客户端在经过了 2MSL 一段时间后自动进入了 CLOSE 状态至此客户端也完成连接的关闭 每一个方向都需要一个 FIN 和一个 ACK因此通常被称为四次挥手。 主动关闭连接的才会有 TIME_WAIT 状态。 1.3.2、为什么回收需要四次 关闭连接时客户端向服务端发送 FIN 时仅仅表示客户端不再发送数据了但是还能接收数据服务端收到客户端的 FIN 报文时先回一个 ACK 应答报文而服务端可能还有数据需要处理和发送等服务端不再发送数据时才发送 FIN 报文给客户端来表示同意现在关闭连接。 但是在特定情况下四次挥手是可以变成三次挥手的。 1.3.3、第一次挥手丢失了会发生什么 当客户端主动关闭方调用 close 函数之后就会向服务端发送 FIN 报文试图与服务端断开连接此时客户端的连接进入到 FIN_WAIT_1 状态。 正常情况下如果能及时收到服务端被动关闭方的 ACK 则会很快变为 FIN_WAIT2 状态。 如果第一次回收丢失了那么客户端迟迟收不到被动方的 ACK 的话也就会触发超时重传机制重传 FIN 报文重发次数由 tcp_orphan_retries 参数控制。 当客户端重传 FIN 报文的次数超过 tcp_orphan_retries 后就不再发送 FIN 报文则会在等待一段时间如果还是没能收到第二次挥手那么直接进入到 close 状态。举个 假设 tcp_orphan_retries 参数值设置为 3 具体过程 当客户端超时重传 3 次 FIN 报文后由于 tcp_orphan_retries 为 3已达到最大重传次数于是再等待一段时间如果还是没能收到服务端的第二次挥手那么客户端就会断开连接。 1.3.4、第二次挥手丢失了会发生什么 当服务器收到客户端的第一次挥手后就会先回一个 ACK 确认报文此时服务端的连接进入到 CLOSE_WAIT 状态。 在前面也提到过ACK 报文是不会重传的所以如果服务端的第二次挥手丢失了客户端就会触发超时重传机制重传 FIN 报文直到收到服务端的第二次挥手或者达到最大的重传次数。 举个假设 tcp_orphan_retries 参数值设置为 2当第二次挥手一直丢失时发生的过程如下 具体过程 当客户端超时重传 2 次 FIN 报文后由于 tcp_orphan_retries 为 2 已达到最大重传次数于是再等待一段时间如果还是没能收到服务端的第二次挥手那么客户端就会断开连接。 此处提一下当客户端收到第二次挥手也就是收到服务端发送的 ACK 报文之后客户端就会处于 FIN_WAIT2 状态在这个状态需要等服务端发送第三次挥手也就是服务端的 FIN 报文。 对于 close 函数关闭的连接由于无法再发送和接收数据所以 FIN_WAIT2 状态不能持续太久而 tcp_fin_timeout 控制了在这个状态下连接的持续时长默认值是 60s。 这就意味着对于调用 close 关闭的连接如果在 60s 后还没有收到 FIN 报文客户端的连接就会直接关闭如下图 但是注意如果主动关闭方使用 shutdown 函数关闭连接指定了只关闭发送方向而接收方向并没有关闭那么意味着主动关闭方还是可以接受数据的。此时如果主动关闭方一直没收到第三次挥手那么主动关闭方的连接将会一直处于 FIN_WAIT2 状态tcp_fin_timeout 无法控制 shutdown 关闭的连接。如下图 1.3.5、第三次挥手丢失了会发生什么 当服务端收到客户端的 FIN 报文后内核会自动回复 ACK同时连接处于 CLOSE_WAIT 状态它表示等待应用进程调用 close 函数关闭连接。 此时内核是没有权利替代进程关闭连接必须由进程主动调用 close 函数来触发服务端发送 FIN 报文。 服务端处于 CLOSE_WAIT 状态时调用了 close 函数内核就会发出 FIN 报文同时连接进入 LAST_ACK 状态等待客户端返回 ACK 来确认连接关闭。 如果迟迟收不到这个 ACK 服务端就会重发 FIN 报文重发次数仍然由 tcp_orphan_retries 参数控制这与客户端重发 FIN 报文的重传次数控制方法是一致的。 举个假设 tcp_orphan_retrie 3当第三次挥手一直丢失时 具体过程 当服务端重传第三次挥手报文的参数达到了 3 次之后由于 tcp_orphan_retries 为 3达到了重传最大次数于是再等待一段时间如果还是没能收到客户端的第四次挥手那么服务端就会断开连接。客户端因为是通过 close 函数关闭连接的处于 FIN_WAIT_2 状态是有时长限制的如果 tcp_fin_timeout 时间内还是没能收到服务端的第三次挥手那么客户端就会断开连接。 1.3.6、如果第四次挥手丢失了会怎么办 当客户端收到服务端的第三次挥手的 FIN 报文后就会回 ACK 报文也就是第四次挥手此时客户端连接进入 TIME_WAIT 状态。 在 Linux 系统TIME_WAIT 状态会持续 2 MSL 后才会进入关闭状态。如果第四次挥手的 ACK 报文没有到达服务端服务端就会重发 FIN 报文重发次数仍然由前面介绍的 tcp_orphan_retries 参数控制。 举个假设 tcp_orphan_retries 参数为 2 当第四次挥手一直丢失时过程如下 具体过程 当服务端重传第三次挥手报文达到 2 时由于 tcp_orphan_retries 为 2 达到了最大重传次数于是再等待一段时间如果还是没能收到客户端的第四次挥手那么服务端就会断开连接。 客户端在收到第三次挥手后就会进入 TIME_WAIT 状态开启时长为 2MSL 的定时器如果途中再次收到第三次挥手后就会重置定时器当等待 2MSL 时长后客户端就会关闭连接。 1.3.7、为什么 TIME_WAIT 等待的时间是 2MSL MSL 是 Maximum Seg Lifetime 报文最大生存时间它是任何报文在网络上存在的最长时间超过这个时间报文将被丢弃因为 TCP 报文基于是 IP 报文的而 IP 头中有一个 TTL 字段是 IP 数据报可以经过的最大路由数每经过一个处理他的路由器此值就减 1当此值为 0 则数据报将被丢弃同时发送 ICMP 报文通知源主机。 MSL 与 TTL 的区别MSL 的单位是时间而 TTL 则是经过路由跳数。所以 MSL 应该要大于等于 TTL 消耗的时间为 0 的时间以确保报文已被自然消灭。 TTL 的值一般为 64Linux 将 MSL 设置为 30s 意味着 Linux 认为数据报文经过 64 个路由器的时间不会超过 30s 如果超过了就认为报文已经消失在网络中了。 TIME_WAIT 等待 2 倍的 MSL 比较合理的解释是网络中可能存在来自发送方的数据包当这些发送方的数据包被接收方处理后又会向对方发送响应所以一来一回需要等待 2 倍的时间。 比如。如果被动关闭方没有收到断开连接的最后的 ACK 报文就会触发超时重发 FIN 报文另一方接收到 FIN 后会重发 ACK 给被动方一来一去正好 2 个 MSL。 可以看到这 2MSL时长 这其实是相当于至少允许报文丢失一次。比如若 ACK 在一个 MSL 内丢失这样被动方重发的 FIN 会在第 2 个 MSL 内到达TIME_WAIT 状态的连接可以应对。 为什么不是 4 或者 8 MSL 时长呢可以想象一个丢包率达到百分之一的糟糕网络连续两次丢包的概率只有万分之一这个概率实在太小了忽略它比解决它更具性价比。 2MSL 的时间是从客户端接收到 FIN 后发送 ACK 开始计时的。如果在 TIME_WAIT 时间内因为客户端的 ACK 没有传输到服务端客户端又接收到了服务端重发的 FIN 报文那么 2 MSL 时间将要重新计时。 在 Linux 系统里 2MSL 默认是 60s 那么一个 MSL 也就是 30sLinux 系统的停留在 TIME_WAIT 的时间为固定的 60s。定义在 Linux 内核代码里的名称为 TCP_TIMEWAIT_LEN #define TCP_TIMEWAITLEN (60*HZ) 如果要修改 TIME_WAIT 的时间长度只能修改 Linux 内核代码里 TCP_TIMEWAIT_LEN 的值并重新编译 Linux 内核。 1.3.8、为什么需要 TIME_WAIT 状态 主动发起关闭连接的一方才会有 TIME-WAIT 状态 需要 TIME-WAIT 状态主要是两个原因 防止历史连接中的数据被后面相同四元组的连接错误的接收保证【被动关闭连接】的一方能够被正常的关闭 ①、防止历史连接数据中的数据被后面相同的四元组的连接错误的接收 为了更好地理解这个原因我们先来了解序列号SEQ和初始序列号ISN。 序列号是 TCP 一个头部字段标识了 TCP 发送端到 TCP 接收端的数据流的一个字节因为 TCP 是面向字节流的可靠传输协议为了保证消息的顺序性和可靠性TCP 为每个传输方向上的每个字节都赋予了一个编号以便于传输成功后确认、丢失后重传以及在接收端保证不会乱序。序列号是一个 32 位的无符号数因此在达到 4G 之后再循环回到 0 。 初始序列号在 TCP 建立连接的时候客户端和服务端都会各自生成一个初始序列号它是基于时钟生成的一个随机数来保证每个连接都拥有不同的初始序列号。初始序列号可被视为一个 32 位的计数器该计数器的数值每 4 微妙加 1循环一次需要 4.55 小时。 通过前面我们知道序列号和初始化序列号并不是无限递增的会发生回绕为初始值的情况这意味着无法根据序列号来判断新老数据。 假设 TIME-WAIT 没有等待时间或时间过短被延迟的数据包抵达后会发生什么呢 如上图 服务端在关闭连接之前发送的 SEQ 301 报文被网络延迟了 接着服务端以相同的四元组重新打开了新连接前面被延迟的 SEQ 301 这时抵达了客户端而且该数据报文的序列号刚好在客户端的接收窗口内因此客户端会正常接收这个数据报文但是这个数据报文是上一个连接残留下来的这样就会产生数据错乱的问题。 为了防止历史连接中的数据被后面相同的四元组的连接错误的接收因此 TCP 设计了 TIME_WAIT 状态状态会持续 2MSL 时长这个时间足以让两个方向上的数据包都会被丢弃使得原来连接的数据包在网络中都消失再出现的数据包一定都是新建立连接所产生的。 ②、保证【被动关闭连接】的一方能被正确的关闭 等待足够的时间以确保最后的 ACK 能让被动方关闭方接收从而帮助其正常关闭。 如果客户端主动关闭方最后一次 ACK 报文在网络中丢失了那么按照 TCP 可靠性原则服务端会重发 FIN 报文。假设客户端没有 TIME_WAIT 状态而是在发完最后一次回 ACK 报文就直接进入 CLOSE 状态 如果该 ACK 报文丢失了服务端则重传的 FIN 报文而这时客户端已经进入到关闭状态了在收到服务端重传的 FIN 报文后就会回 RST 报文。如图 服务端收到这个 RST 并将其解释为一个错误Connection reset by peer这对于一个可靠的协议来说不是一个优雅的终止方式。为了防止这种情况出现客户端必须等待足够长的时间确保服务端能够收到 ACK 如果服务端没有收到 ACK 那么就会触发 TCP 重传机制服务端会重新发送一个 FIN 这样一去一来刚好两个 MSL 的时间。 1.3.9、TIME_WAIT 过多有什么危害 主要有两种 占用系统资源比如文件描述符内存资源CPU资源线程资源等 占用端口资源端口资源也是有限的一般可以开启的端口是 32768~61000也可以通过 net.ipv4.ip_local_port_range 参数指定范围。客户端和服务端 TIME_WAIT 过多造成的影响是不同的。 如果客户端主动发起关闭连接的一方的 TIME_WAIT 状态过多占满了所有端口资源那么就无法对【目的 IP 目的 PORT】都一样的服务端发起连接了但是被使用的端口还是可以继续对另外的一个服务端发起连接的。 因此客户端发起连接方都是和【目的 IP 目的 PORT】都一样的服务端建立连接的话当客户端的 TIME_WAIT 状态连接过多的话就会受端口资源限制如果沾满了所有端口资源那么就无法再跟【目的端口IP 目的 PORT】都一样的服务端建立连接了。 不过即使在这种场景下只要连接的是不同的服务端端口都是可以重复使用的所以客户端还是可以向其他服务端发起连接的这是因为内核在定位一个连接的时候是通过【四元组】信息来定位的并不会因为客户端的端口一样而导致连接冲突。 如果服务端主动发起关闭连接方的 TIME_WAIT 状态过多并不会导致端口资源受限因为服务端只监听一个端口而且由于一个四元组唯一确认一个 TCP 连接因此理论上服务端可以建立很多连接但是 TCP 连接过多会占用系统资源比如文件描述符内存资源CPU资源线程资源等。 1.3.10、服务器出现大量 TIME_WAIT 状态的原因有哪些 要知道TIME_WAIT 状态是主动连接方才会出现的状态所以如果服务器出现大量的 TIME_WAIT 状态的 TCP 连接就是说明服务器主动断开了很多 TCP 连接。 什么场景下服务端会主动断开连接呢 HTTP 没有使用长连接HTTP 长连接超时HTTP 长连接的请求数量达到上限 ①、HTTP 没有使用长连接 从 HTTP/1.1 开始就默认是开启了 Keep-Alive 只要客户端和服务端任意一方的 HTTP header 中有 Connectionclose 信息那么就无法使用 HTTP 长连接的机制。绝大多数 Web 服务的实现不管哪一方禁用了 HTTP Keep-Alive 都是由服务端主动关闭连接。 客户端禁用了 HTTP Keep-Alive 服务端开启了 HTTP Keep-Alive 谁是主动关闭方 当客户端禁用之后HTTP 请求的 header 就会有 Connectionclose 信息这时候服务端在发送完 HTTP 响应之后就会主动关闭。为什么要这样设计呢HTTP 是请求-响应模型发起方一直是客户端HTTP Keep-Alive 的初衷是为客户后续的请求重用连接如果我们在某次 HTTP 请求-响应模型中请求的 header 定义了 connectionclose 信息那不再重用这个连接的时机就只有在服务端了所以我们在 HTTP 请求-响应这个周期的【末端】关闭连接是合理的。 客户端开启了 HTTP Keep-Alive 服务端禁用了 HTTP Keep-Alive 谁是主动连接方 此时服务端在发完 HTTP 响应之后服务端也会主动关闭连接。为什么这样设计呢 在服务端主动关闭连接的情况下只要调用一次 close() 就可以释放连接 剩下的工作由内核 TCP 栈直接进行了处理整个过程只有一次系统调用syscall如果是客户端要求关闭则服务端在写完最后一个响应response之后需要把这个 socket 放入到 readable 队列调用 select/epoll 去等待事件然后调用一次 read() 才能知道连接已经被关闭这其中是两次系统调用syscall多一次用户态程序被激活执行而且 socket 保持时间也会更长。因此当服务端出线了大量的 TIME_WAIT 状态连接时可以排查下客户端和服务端是否都开起了 Keep-Alive 因为任意一方没有开启的话都会导致服务端在处理完一个 HTTP 请求后就主动关闭连接此时服务端上就会出现大量的 TIME_WAIT 状态的连接。解决办法也很简单客户端服务端都开启 Keep-Alive。 ②、HTTP 长连接超时 HTTP 长连接的特点是只要任意一端没有明确提出断开连接则保持 TCP 连接状态。 HTTP 长连接可以在同一个 TCP 连接上接收和发送多个 HTTP 请求/应答 ,避免了连接建立和释放的开销。 如果使用了 HTTP 长连接如果客户端完成一个 HTTP 请求之后就不再发起新的请求此时这个 TCP 连接一直占用着不是挺浪费资源的吗 所以为了避免资源浪费的情况web 服务软件一般都会提供一个参数用来指定 HTTP 长连接的超时时间比如 nginx 提供的 keepalive_timeout 参数。假如设置了 HTTP 长连接的超时时间是 60s nginx 就会启动一个【定时器】如果客户端在完后一个 HTTP 请求后在 60s 内都没有再发起新的请求定时器的时间一到nginx 就会触发回调函数来关闭该连接那么此时服务端就会出现 TIME_WAIT 状态的连接。 当服务端出现大量的 TIME_WAIT 状态的连接时如果现象是有大量的客户端建立完 TCP 连接后很长一段时间没有发送数据那么大概率是因为 HTTP 长连接超时导致服务端主动关闭连接产生大量处于 TIME_WAIT 状态的连接。或者往网络方面排查是否是因为网络的问题导致客户端发送的数据一直没有被服务端接收以至于 HTTP 长连接超时。 ③、HTTP 长连接的请求数量达到上限 web服务端通常会有个参数来定义一条 HTTP 长连接上最大能处理的请求数量当超过最大限制时就会主动关闭连接。这个参数的默认值是 100意味着每个 HTTP 长连接最多只能跑 100 次请求这个参数往往会被很多很忽略因为当 QPS每秒请求数不是很高时默认值 100 够用。但是对于一些 QPS 比较高的场景比如超过 10000 QPS 甚至达到更高如果此参数还为 100服务端就会很频繁的关闭连接那么此时服务端上就会出现大量的 TIME_WAIT 状态。解决方式也很简单调大参数就行。 1.3.11、服务器出现大量 CLOSE_WAIT 状态的原因有哪些 CLOSE_WAIT 状态是【被动关闭方】才会有的状态而且如果【被动关闭方】没有调用 close 函数关闭连接那么就无法发出 FIN 报文从而使得 CLOSE_WAIT 状态的连接转变为 LAST_ACK 状态。 所以当服务端出现大量 CLOSE_WAIT 状态的连接时说明服务端的程序没有调用 close 函数关闭连接。 先来分析一下一个普通的 TCP 服务端的流程 创建服务端 socket bind 绑定窗口listen 监听窗口将服务端 socket 注册到 epollepoll_wait 等待连接到来连接到来时调用 accpet 获取已连接的 socket将已连接的 socket 注册到 epollepoll_wait 等待事件发生对方关闭连接时我方调用 close 可能导致没有调用 close 函数的原因 第 2 步没有做没有将服务端 socket 注册到 epoll这样有新连接到来时服务端没办法感知这个事件也就无法获取已连接的 socket 那服务端自然就没机会对 socket 调用 close 函数。第 3 步没有做有新连接到来时没有调用 accpet 获取该连接的 socket 导致有大量的客户端主动断开了连接而服务端没机会对这些 socket 调用 close 函数导致服务端出现大量的 CLOSE_WAIT 状态的连接。可能出现在服务端运行 accept 函数之前代码卡在某一逻辑或者提前抛出了错误。第 4 步没有做通过 accept 获取已连接的 socket 后没有将其注册到 epoll 导致后续收到 FIN 报文时服务端没办法感知这个事件那服务端就没机会调用 close 函数了。第 6 步没有做当发现客户端关闭连接后服务端没有执行 close 函数可能是因为代码泄露或者是在执行 close 函数之前代码卡在某一逻辑比如发生死锁。 1.3.12、如果已经建立连接但是客户端突然发生故障怎么办 客户端出现故障是指客户端的主机发生了宕机或者断电的场景。发生这种情况时如果服务端一直不会发送数据给客户端那么服务端是永远无法感知到客户机宕机这个事件的也就是服务端的 TCP 连接将一直处于 ESTABLISH 状态占用着系统资源。为了避免这个情况TCP 有保活机制定义一个时间段在这个时间段内如果没有任何连接相关的活动TCP 保活机制就会开始作用每隔一个时间间隔发送一个探测报文该探测报文包含的数据非常少如果连续几个探测报文都没有得到响应则认为当前的 TCP 连接已死亡系统内核将错误信息通知上层应用。在 Linux 内核可以有对应的参数可以设置保活时间保活探测次数和间隔时间以下为默认值 net.ipv4.tcp_keepalive_time7200 net.ipv4.tcp_keepalive_intvl75 net.ipv4.tcp_keepalive_probes9 tcp_keepalive_time7200表示保活时间是 7200 秒2小时也就 2 小时内如果没有任何连接相关的活动则会启动保活机制tcp_keepalive_intvl75表示每次检测间隔 75 秒tcp_keepalive_probes9表示检测 9 次无响应认为对方是不可达的从而中断本次的连接。 也就是说在 Linux 系统中最少需要 2 小时 11 分 15秒 才可以检测出一个【死亡】连接。 应用程序若想使用保活机制则需要通过 socket 接口设置 SO_KEEPALIVE 选项才生效。 开启之后会遇到这几种情况 对端程序是正常工作的当 TCP 保活的探测报文发送给对端对端会正常响应这样 TCP 保活时间会重置。 对端主机宕机并重启当 TCP 保活的探测报文发送给对端后对端可以响应但由于没有该连接的有效信息会产生一个 RST 报文这样很快就发现 TCP 连接已被重置。 对端主机宕机注意不是进程崩溃进程崩溃后操作系统在回收进程资源时会发送 FIN 报文而主机宕机则是无法感知的所以需要 TCP 保活机制来探测对方是不是发生了主机宕机或对端由于其他原因导致报文不可达。当 TCP 保活的探测报文发送后达到保活探测次数后TCP 会报告该连接已死亡。 这个 TCP 的保活机制有点长我们可以在应用层实现一个心跳机制 比如 web 服务软件一般都提供 keepalive_timeout 参数用来指定 HTTP 长连接的超时时间如果设置了 HTTP 长连接的超时时间为 60s web 服务端软件就会启动一个定时器如果客户端在完成一个 HTTP 请求后在 60s 内都没有再发起新的请求定时器时间一到就会触发回调函数来释放该连接。 1.3.13、如果已经建立连接但是服务端的进程崩溃会发生什么 TCP 的连接信息是由内核维护的所以当服务端的进程崩溃后内核需要回收该进程的所有 TCP 连接资源于是内核就会发送第一次挥手 FIN 报文后续的挥手过程也都是在内核完成并不需要进程的参与所以即使服务端的进程退出了还是能与客户端完成 TCP 四次挥手的过程。 1.4、Socket 编程 1.4.1、针对 TCP 应该如何 Socket 编程 服务端和客户端初始化 socket得到文件描述符服务端调用 bind 将 socket 绑定在指定的 IP 地址和端口服务端调用 listen进行监听服务端调用 connect向服务端的地址和端口发起连接请求服务端 accept 返回用于传输的 socket 的文件描述符客户端调用 write 写入数据服务端调用 read 读取数据客户端断开连接后会调用 close那么服务端 read 读取数据的时候就会读到 EOF 待处理完数据之后服务端调用 close表示连接关闭 这里需要注意的是服务端调用 accept 时连接成功了会返回一个已完成连接的 socket 后续用来传输数据。 所以监听的 socket 和真正用来传送数据的 socket 是【两个】socket一个叫做监听 socket一个叫做已完成连接 socket。 1.4.2、listen 参数 backlog 的意义 Linux 内核中会维护两个队列 半连接队列SYN 队列接收到一个 SYN 建立连接请求处于 SYN_RCVD 状态全连接队列Accept 队列已完成 TCP 三次握手过程处于 ESTABLISHED 状态 如图 int listen (int socketfd, int backlog) 参数一socketfd 为 socketfd 文件描述符参数二backlog 这参数在历史版本有一定的变化 在早期的 Linux 内核中backlog 变成 accept 队列也就是已完成连接建立的队列长度所以现在通常认为 backlog 是 accept 队列。但是上限值是内核参数 somaxconn 的大小也就是说 accept 队列长度 min(backlog , somaxconn)。 1.4.3、accept 发生在三次握手的哪一步 如图 客户端的协议栈向服务端发送了 SYN 包并告诉服务端当前发送序列号 client_isn 客户端进入到 SYN_WAIT 状态服务端的协议栈收到这个包之后和客户端进行 ACK 应答应答的值为 client_isn 1表示对 SYN 包 client_isn 的确认同时服务端也发送一个 SYN 包告诉客户当前我的发送序列号为 server_isn 服务端进入 SYN_RCVD 状态客户端协议栈收到 ACK 之后使得应用程序从 connect 调用返回表示客户端到服务端的单向连接建立成功客户端的状态为 ESTABLISHED 同时客户端协议栈也会对服务端的 SYN 包进行应答应答数据为 server_isn 1 ACK 应答包达到服务端后服务端的 TCP 连接进入 ESTABLISHED 状态同时服务端协议栈使得 accept 阻塞调用返回这个时候服务端到客户端的单项连接也建立成功。至此客户端与服务端两个方向的连接都建立成功。 我们可以得知客户端 connect 成功返回是在第二次握手服务端 accept 成功返回是在三次握手成功之后。 1.1.4、客户端调用 close 了连接是断开的流程是什么 咱们看看客户端主动调用 close 会发生什么 客户端调用 close 表明客户端没有数据要发送了则此时会向服务端发送 FIN 报文进入 FIN_WAIT_1 状态服务端接收到了 FIN 报文TCP 协议栈会为 FIN 包插入一个文件结束符 EOF 到接收缓冲区中应用程序可以通过 read 调用来感知这个 FIN 包。这个 EOF 会被放在已排队等待的其他已接收的数据之后这就意味着服务端需要处理这种异常情况因为 EOF 表示在该连接上再无额外数据到达。此时服务端进入 CLOSE_WAIT 状态接着当处理完数据后自然就会读到 EOF于是也调用 close 关闭它的套接字这会使得服务端发出一个 FIN 包之后处于 LAST_ACK 状态客户端收到服务端的 FIN 包并发送 ACK 确认包给服务端此时客户端将进入 TIME_WAIT 状态服务端收到 ACK 确认包之后就进入到了最后的 CLOSE 状态客户端经过 2MSL 时间之后也进入 CLOSE 状态 1.1.5、没有 accept 能建立 TCP 连接吗 可以的。accept 系统调用并不参与 TCP 三次握手过程它只是负责从 TCP 全连接队列中取出一个已经建立连接的 socket 用户层通过 accept 系统调用拿到了已经建立连接的 socket 就可以对该 socket 进行读写操作了。 1.1.6、没有 listen能建立 TCP 连接吗 可以的客户端是可以自己连自己的形成连接TCP自连接也可以两个客户端同时向对方发出请求建立连接TCP 同时打开这两个情况都有个共同点就是没有服务端参与也就是没有 listen 就能 TCP 建立连接。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/922900.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

23种设计模式——组合模式(Composite Pattern) - 指南

pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", …

网站开发报价单 excel曰本做爰l网站

文档介绍:6.5 算术逻辑单元一、ALU 电路四位 ALU 74181M 0 算术运算M 1 逻辑运算S3 ~ S0 不同取值,可做不同运算ALUAiBiFi…SiM位片式运算器4位双极型位片式运算器AM2901,它将ALU、通用寄存器组、多路开关、移位器等逻辑构件集成在一个芯片内。通用寄存器寄存器写…

2025年破碎机厂家最新权威推荐榜:破碎机实力厂商技术服务全景评测及选购指南

在矿山开发、冶金加工、建材生产等核心领域,破碎机作为关键生产装备,其性能与服务直接决定项目效率、资源利用率及运营成本。当前市场中,破碎机品牌超千家,产品质量与服务能力却存在显著差距:部分厂商缺乏核心技术…

网站做发做网站站长先把作息和身体搞好

当某些sql因为不知名原因堵塞时,为了不影响后台服务运行,想要给sql增加执行时间限制,超时后就抛异常,保证后台线程不会因为sql堵塞而堵塞。 方法一 yml全局配置:单数据源可以,多数据源时会失效 方法二 j…

什么关系?就是ajax与jQuery

什么关系?就是ajax与jQuerypre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco",…

网站建设过时了吗没有网站没有推广如何做外贸

目录 1 创建购物车2 增加数量3 减少数量4 切换分类时回填数据5 显示购物车信息总结 我们上一篇搭建了点餐业务的数据初始化加载,本篇实现一下加入购物车的功能。在购物车设计的时候有两种方案,一种是使用数据表的方案,一种是使用变量的方案。…

完整教程:Redis数据结构和常用命令

pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", …

【光照】[PBR][法线分布]为何不选Beckmann

本文探讨了Beckmann分布函数在游戏渲染中的应用及其与GGX的对比。Beckmann是最早的微表面法线分布函数,基于高斯分布假设,具有物理准确性但计算复杂度较高。GGX因更符合真实材质反射特性、能量守恒良好、计算效率更高…

网站导航页面制作网站宣传页面

题目 给定一个长度为 n1 的数组nums,数组中所有的数均在 1∼n 的范围内,其中 n≥1。请找出数组中任意一个重复的数,但不能修改输入的数组。样例 给定 nums [2, 3, 5, 4, 3, 2, 6, 7]。 返回 2 或 3。思考题: 如果只能使用 O(1) …

钓鱼网站怎么做的专做衬衫的网站

英语自我介绍,介绍完老师会根据你的回答用英语问你问题,比如介绍一下你的本科学校,或者家乡什么的。计网过一遍,会问两道题。接下来是重点,我当时是根据我成绩单,问了我本科学过的科目,比如pyth…

2025标志牌生产厂家最新推荐排行榜:权威筛选优质标志牌品牌,助您精准选对交通标志牌,反光标志牌,道路标志牌供应商!

随着道路建设与交通管理需求的不断提升,标志牌作为保障通行安全、传递交通信息的核心设施,市场需求持续增长。但当前标志牌生产行业中,厂家数量繁杂,产品质量差距悬殊 —— 部分产品存在铝板厚度不达标、反光膜亮度…

2025 年脚手架厂家最新推荐榜:铝合金 / 盘扣 / 快装 / 移动式等多类型产品优选及国内实力企业排行指南

2025 年国内脚手架行业在基建加码与装配式建筑发展中持续扩容,但市场痛点凸显:传统扣件式产品仍占 42% 份额,安全与效率短板显著,钢管价格上涨进一步压缩利润空间,新国标下部分产品合规性不足。更关键的是,近五年…

清远网站seo公司网站建设平台策划

Hello大家好我是咕噜铁蛋!你是否曾为生成二维码而烦恼过?别担心,今天我就来给你支招!,我将分享如何使用Java生成微信小程序二维码的方法,让你轻松应对二维码生成的需求。微信小程序是一种轻量级的应用程序&…

2025年沈阳标识标牌厂家最新推荐榜单:涵盖订做标识标牌,广告标识标牌,安全出口标识标牌、不锈钢等多类型标识,全面解读企业产能与技术实力

在城市建设加速、商业场景多元化的当下,标识标牌作为信息传递与环境导向的核心载体,其品质与服务直接影响项目落地效果。然而,当前市场上标识标牌厂家数量繁杂,部分企业存在产能不足、技术落后、设计同质化等问题,…

网站开发研究的方法与技术路线网络营销渠道的功能

哥伦比亚大学实力介绍哥伦比亚大学除去常青藤的榜首地位,研究生的金融工程 更是被人们称为是金工届的magic seven 并且商学院以及MBA项目是很多人垂涎已久的。更有远近闻名的国际公共关系事物学院,以及新闻专业。2020年TFE Times美国计算机科学硕士排名参…

别再迷信甘特图了!90%的项目用它都错了

甘特图被90%的项目团队误用,导致其真正的管理价值被严重低估。作为最经典的项目规划工具,甘特图本应通过时间轴可视化、任务依赖关系展示和资源分配监控来提升协作效率,但多数使用者陷入七个致命误区:将其视为静态…

c2c网站建设费用福建省建设厅网站人员

图像到卡通 一、说明 在当今世界,我们被图像和视频所包围。从社交媒体到广告,图像已成为一种强大的交流媒介。但是你有没有想过,如果你能把你的照片变成卡通会发生什么?想象一下,为您最喜欢的照片创建动画版本&#xf…

手机网站建设流程国际新闻最近新闻军事

Zabbix核心组件主要是Agent和Server,其中Agent主要负责采集数据并通过主动或者被动的方式采集数据发送到Server/Proxy,除此之外,为了扩展监控项,Agent还支持执行自定义脚本。 作者:艺术生的运维路来源:今日…

浙江建设集团网站手机网站建设找哪家

在内核编程中字符串有两种格式ANSI_STRING与UNICODE_STRING,这两种格式是微软推出的安全版本的字符串结构体,也是微软推荐使用的格式,通常情况下ANSI_STRING代表的类型是char *也就是ANSI多字节模式的字符串,而UNICODE_STRING则代…

廉政建设网评文章网站直播网站app下载

介绍 假设现有外网笔记本、云服务器、内网工作站三台设备,希望使用外网笔记本通过云服务器转发,访问内网工作站;这里使用frp进行内网穿透。 云服务器端配置 登录腾讯轻量型云服务器控制台,开放转发端口、bind_port以及deshboad…