TCP 四次挥手中,能不能把第二次的 ACK 报文, 放到第三次 FIN 报文一起发送? 虽然我们在学习 TCP 挥手时,学到的是需要四次来完成 TCP 挥手,但是 在一些情况下, TCP 四次挥手是可以变成 TCP 三次挥手的而且在用 wireshark 工具抓包的时候,我们也会常看到 TCP 挥手过程是三次,而不是四次,如下图: 先来回答为什么 RFC 文档里定义 TCP 挥手过程是要四次?再来回答什么情况下,什么情况会出现三次挥手?

为什么 TCP 挥手需要四次?

TCP 四次挥手的过程如下: 具体过程:

  • 客户端主动调用关闭连接的函数,于是就会发送 FIN 报文,这个  FIN 报文代表客户端不会再发送数据了,进入 FIN_WAIT_1 状态;
  • 服务端收到了 FIN 报文,然后马上回复一个 ACK 确认报文,此时服务端进入 CLOSE_WAIT 状态。在收到 FIN 报文的时候,TCP 协议栈会为 FIN 包插入一个文件结束符 EOF 到接收缓冲区中,服务端应用程序可以通过 read 调用来感知这个 FIN 包,这个 EOF 会被放在已排队等候的其他已接收的数据之后,所以必须要得继续 read 接收缓冲区已接收的数据;
  • 接着,当服务端在 read 数据的时候,最后自然就会读到 EOF,接着 read() 就会返回 0,这时服务端应用程序如果有数据要发送的话,就发完数据后才调用关闭连接的函数,如果服务端应用程序没有数据要发送的话,可以直接调用关闭连接的函数,这时服务端就会发一个 FIN 包,这个  FIN 报文代表服务端不会再发送数据了,之后处于 LAST_ACK 状态;
  • 客户端接收到服务端的 FIN 包,并发送 ACK 确认包给服务端,此时客户端将进入 TIME_WAIT 状态;
  • 服务端收到 ACK 确认包后,就进入了最后的 CLOSE 状态;
  • 客户端经过 2MSL 时间之后,也进入 CLOSE 状态;

你可以看到,每个方向都需要 一个 FIN 和一个 ACK,因此通常被称为 四次挥手

为什么 TCP 挥手需要四次呢?

服务器收到客户端的 FIN 报文时,内核会马上回一个 ACK 应答报文, 但是服务端应用程序可能还有数据要发送,所以并不能马上发送 FIN 报文,而是将发送 FIN 报文的控制权交给服务端应用程序

  • 如果服务端应用程序有数据要发送的话,就发完数据后,才调用关闭连接的函数;
  • 如果服务端应用程序没有数据要发送的话,可以直接调用关闭连接的函数,

从上面过程可知, 是否要发送第三次挥手的控制权不在内核,而是在被动关闭方(上图的服务端)的应用程序,因为应用程序可能还有数据要发送,由应用程序决定什么时候调用关闭连接的函数,当调用了关闭连接的函数,内核就会发送 FIN 报文了,所以服务端的 ACK 和 FIN 一般都会分开发送。

FIN 报文一定得调用关闭连接的函数,才会发送吗?

不一定。如果进程退出了,不管是不是正常退出,还是异常退出(如进程崩溃),内核都会发送 FIN 报文,与对方完成四次挥手。

粗暴关闭 vs 优雅关闭

前面介绍 TCP 四次挥手的时候,并没有详细介绍关闭连接的函数,其实关闭的连接的函数有两种函数:

  • close 函数,同时 socket 关闭发送方向和读取方向,也就是 socket 不再有发送和接收数据的能力。如果有多进程/多线程共享同一个 socket,如果有一个进程调用了 close 关闭只是让 socket 引用计数 -1,并不会导致 socket 不可用,同时也不会发出 FIN 报文,其他进程还是可以正常读写该 socket,直到引用计数变为 0,才会发出 FIN 报文。
  • shutdown 函数,可以指定 socket 只关闭发送方向而不关闭读取方向,也就是 socket 不再有发送数据的能力,但是还是具有接收数据的能力。如果有多进程/多线程共享同一个 socket,shutdown 则不管引用计数,直接使得该 socket 不可用,然后发出 FIN 报文,如果有别的进程企图使用该 socket,将会受到影响。

如果客户端是用 close 函数来关闭连接,那么在 TCP 四次挥手过程中,如果收到了服务端发送的数据,由于客户端已经不再具有发送和接收数据的能力,所以客户端的内核会回 RST 报文给服务端,然后内核会释放连接,这时就不会经历完成的 TCP 四次挥手,所以我们常说,调用 close 是粗暴的关闭。 当服务端收到 RST 后,内核就会释放连接,当服务端应用程序再次发起读操作或者写操作时,就能感知到连接已经被释放了:

  • 如果是读操作,则会返回 RST 的报错,也就是我们常见的Connection reset by peer。
  • 如果是写操作,那么程序会产生 SIGPIPE 信号,应用层代码可以捕获并处理信号,如果不处理,则默认情况下进程会终止,异常退出。

相对的,shutdown 函数因为可以指定只关闭发送方向而不关闭读取方向,所以即使在 TCP 四次挥手过程中,如果收到了服务端发送的数据,客户端也是可以正常读取到该数据的,然后就会经历完整的 TCP 四次挥手,所以我们常说,调用 shutdown 是优雅的关闭。 但是注意,shutdown 函数也可以指定「只关闭读取方向,而不关闭发送方向」,但是这时候内核是不会发送 FIN 报文的,因为发送 FIN 报文是意味着我方将不再发送任何数据,而  shutdown 如果指定「不关闭发送方向」,就意味着 socket 还有发送数据的能力,所以内核就不会发送 FIN

什么情况会出现三次挥手?

当被动关闭方(上图的服务端)在 TCP 挥手过程中,「 没有数据要发送」并且「开启了 TCP 延迟确认机制」,那么第二和第三次挥手就会合并传输,这样就出现了三次挥手。 然后因为 TCP 延迟确认机制是默认开启的,所以导致我们抓包时,看见三次挥手的次数比四次挥手还多。

什么是  TCP 延迟确认机制?

当发送没有携带数据的 ACK,它的网络效率也是很低的,因为它也有 40 个字节的 IP 头 和 TCP 头,但却没有携带数据报文。为了解决 ACK 传输效率低问题,所以就衍生出了  TCP 延迟确认。TCP 延迟确认的策略:

  • 当有响应数据要发送时,ACK 会随着响应数据一起立刻发送给对方
  • 当没有响应数据要发送时,ACK 将会延迟一段时间,以等待是否有响应数据可以一起发送
  • 如果在延迟等待发送 ACK 期间,对方的第二个数据报文又到达了,这时就会立刻发送 ACK

延迟等待的时间是在 Linux 内核中定义的,如下图:

关键就需要 HZ 这个数值大小,HZ 是跟系统的时钟频率有关,每个操作系统都不一样,在我的 Linux 系统中 HZ 大小是 1000,如下图: 知道了 HZ 的大小,那么就可以算出:

  • 最大延迟确认时间是 200 ms (1000/5)
  • 最短延迟确认时间是 40 ms (1000/25)

怎么关闭 TCP 延迟确认机制?

如果要关闭 TCP 延迟确认机制,可以在 Socket 设置里启用 TCP_QUICKACK,启用  TCP_QUICKACK,就相当于关闭 TCP 延迟确认机制。// 1 表示开启 TCP_QUICKACK,即关闭 TCP 延迟确认机制
int value =  1;
setsockopt(socketfd, IPPROTO_TCP, TCP_QUICKACK, ( char*)& value,  sizeof( int));

实验验证

实验一

接下来,来给大家做个实验,验证这个结论:

当被动关闭方(上图的服务端)在 TCP 挥手过程中,「没有数据要发送」并且「开启了 TCP 延迟确认机制」,那么第二和第三次挥手就会合并传输,这样就出现了三次挥手。

服务端的代码如下,做的事情很简单,就读取数据,然后当 read 返回 0 的时候,就马上调用 close 关闭连接。因为 TCP 延迟确认机制是默认开启的,所以不需要特殊设置。#include&nbsp;<stdlib.h>
#include&nbsp;<stdio.h>
#include&nbsp;<errno.h>
#include&nbsp;<string.h>
#include&nbsp;<netdb.h>
#include&nbsp;<sys/types.h>
#include&nbsp;<netinet/in.h>
#include&nbsp;<sys/socket.h>
#include&nbsp;<netinet/tcp.h>

#define&nbsp;MAXLINE&nbsp;1024

int&nbsp;main(int&nbsp;argc,&nbsp;char&nbsp;*argv[])
{

&nbsp;&nbsp;&nbsp;&nbsp; //&nbsp;1.&nbsp;创建一个监听&nbsp;socket
&nbsp;&nbsp;&nbsp;&nbsp; int&nbsp;listenfd&nbsp;=&nbsp;socket(AF_INET,&nbsp;SOCK_STREAM,&nbsp; 0);
&nbsp;&nbsp;&nbsp;&nbsp; if(listenfd&nbsp;<&nbsp; 0)
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr,&nbsp; "socket&nbsp;error&nbsp;:&nbsp;%s\n",&nbsp;strerror(errno));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return&nbsp; -1;
&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp; //&nbsp;2.&nbsp;初始化服务器地址和端口
&nbsp;&nbsp;&nbsp;&nbsp; struct&nbsp;sockaddr_in&nbsp;server_addr;
&nbsp;&nbsp;&nbsp;&nbsp;bzero(&server_addr,&nbsp; sizeof(struct&nbsp;sockaddr_in));
&nbsp;&nbsp;&nbsp;&nbsp;server_addr.sin_family&nbsp;=&nbsp;AF_INET;
&nbsp;&nbsp;&nbsp;&nbsp;server_addr.sin_addr.s_addr&nbsp;=&nbsp;htonl(INADDR_ANY);
&nbsp;&nbsp;&nbsp;&nbsp;server_addr.sin_port&nbsp;=&nbsp;htons( 8888);

&nbsp;&nbsp;&nbsp;&nbsp; //&nbsp;3.&nbsp;绑定地址+端口
&nbsp;&nbsp;&nbsp;&nbsp; if(bind(listenfd,&nbsp;(struct&nbsp;sockaddr&nbsp;*)(&server_addr),&nbsp; sizeof(struct&nbsp;sockaddr))&nbsp;<&nbsp; 0)
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr, "bind&nbsp;error:%s\n",&nbsp;strerror(errno));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return&nbsp; -1;
&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp; printf( "begin&nbsp;listen....\n");

&nbsp;&nbsp;&nbsp;&nbsp; //&nbsp;4.&nbsp;开始监听
&nbsp;&nbsp;&nbsp;&nbsp; if(listen(listenfd,&nbsp; 128))
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr,&nbsp; "listen&nbsp;error:%s\n\a",&nbsp;strerror(errno));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; exit( 1);
&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp; //&nbsp;5.&nbsp;获取已连接的socket
&nbsp;&nbsp;&nbsp;&nbsp; struct&nbsp;sockaddr_in&nbsp;client_addr;
&nbsp;&nbsp;&nbsp;&nbsp; socklen_t&nbsp;client_addrlen&nbsp;=&nbsp; sizeof(client_addr);
&nbsp;&nbsp;&nbsp;&nbsp; int&nbsp;clientfd&nbsp;=&nbsp;accept(listenfd,&nbsp;(struct&nbsp;sockaddr&nbsp;*)&client_addr,&nbsp;&client_addrlen);
&nbsp;&nbsp;&nbsp;&nbsp; if(clientfd&nbsp;<&nbsp; 0)&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr,&nbsp; "accept&nbsp;error:%s\n\a",&nbsp;strerror(errno));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; exit( 1);
&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp; printf( "accept&nbsp;success\n");

&nbsp;&nbsp;&nbsp;&nbsp; char&nbsp;message[MAXLINE]&nbsp;=&nbsp;{ 0};
&nbsp;&nbsp;&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp; while( 1)&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //6.&nbsp;读取客户端发送的数据
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; int&nbsp;n&nbsp;=&nbsp;read(clientfd,&nbsp;message,&nbsp;MAXLINE);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if(n&nbsp;<&nbsp; 0)&nbsp;{&nbsp; //&nbsp;读取错误
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr,&nbsp; "read&nbsp;error:%s\n\a",&nbsp;strerror(errno));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; break;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp; else&nbsp; if(n&nbsp;==&nbsp; 0)&nbsp;{&nbsp;&nbsp; //&nbsp;返回&nbsp;0&nbsp;,代表读到&nbsp;FIN&nbsp;报文
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr,&nbsp; "client&nbsp;closed&nbsp;\n");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;close(clientfd);&nbsp; //&nbsp;没有数据要发送,立马关闭连接
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; break;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;message[n]&nbsp;=&nbsp; 0;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; printf( "received&nbsp;%d&nbsp;bytes:&nbsp;%s\n",&nbsp;n,&nbsp;message);
&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;close(listenfd);
&nbsp;&nbsp;&nbsp;&nbsp; return&nbsp; 0;
}
客户端代码如下,做的事情也很简单,与服务端连接成功后,就发送数据给服务端,然后睡眠一秒后,就调用 close 关闭连接,所以客户端是主动关闭方:#include&nbsp;<stdlib.h>
#include&nbsp;<stdio.h>
#include&nbsp;<errno.h>
#include&nbsp;<string.h>
#include&nbsp;<netdb.h>
#include&nbsp;<sys/types.h>
#include&nbsp;<netinet/in.h>
#include&nbsp;<sys/socket.h>

int&nbsp;main(int&nbsp;argc,&nbsp;char&nbsp;*argv[])
{

&nbsp;&nbsp;&nbsp;&nbsp; //&nbsp;1.&nbsp;创建一个监听&nbsp;socket
&nbsp;&nbsp;&nbsp;&nbsp; int&nbsp;connectfd&nbsp;=&nbsp;socket(AF_INET,&nbsp;SOCK_STREAM,&nbsp; 0);
&nbsp;&nbsp;&nbsp;&nbsp; if(connectfd&nbsp;<&nbsp; 0)
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr,&nbsp; "socket&nbsp;error&nbsp;:&nbsp;%s\n",&nbsp;strerror(errno));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return&nbsp; -1;
&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp; //&nbsp;2.&nbsp;初始化服务器地址和端口
&nbsp;&nbsp;&nbsp;&nbsp; struct&nbsp;sockaddr_in&nbsp;server_addr;
&nbsp;&nbsp;&nbsp;&nbsp;bzero(&server_addr,&nbsp; sizeof(struct&nbsp;sockaddr_in));
&nbsp;&nbsp;&nbsp;&nbsp;server_addr.sin_family&nbsp;=&nbsp;AF_INET;
&nbsp;&nbsp;&nbsp;&nbsp;server_addr.sin_addr.s_addr&nbsp;=&nbsp;inet_addr( "127.0.0.1");
&nbsp;&nbsp;&nbsp;&nbsp;server_addr.sin_port&nbsp;=&nbsp;htons( 8888);
&nbsp;&nbsp;&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp; //&nbsp;3.&nbsp;连接服务器
&nbsp;&nbsp;&nbsp;&nbsp; if(connect(connectfd,&nbsp;(struct&nbsp;sockaddr&nbsp;*)(&server_addr),&nbsp; sizeof(server_addr))&nbsp;<&nbsp; 0)
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr, "connect&nbsp;error:%s\n",&nbsp;strerror(errno));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return&nbsp; -1;
&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp; printf( "connect&nbsp;success\n");

&nbsp;&nbsp;&nbsp;&nbsp; char&nbsp;sendline[ 64]&nbsp;=&nbsp; "hello,&nbsp;i&nbsp;am&nbsp;xiaolin";

&nbsp;&nbsp;&nbsp;&nbsp; //4.&nbsp;发送数据
&nbsp;&nbsp;&nbsp;&nbsp; int&nbsp;ret&nbsp;=&nbsp;send(connectfd,&nbsp;sendline,&nbsp; strlen(sendline),&nbsp; 0);
&nbsp;&nbsp;&nbsp;&nbsp; if(ret&nbsp;!=&nbsp; strlen(sendline))&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf( stderr, "send&nbsp;data&nbsp;error:%s\n",&nbsp;strerror(errno));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return&nbsp; -1;
&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp; printf( "already&nbsp;send&nbsp;%d&nbsp;bytes\n",&nbsp;ret);

&nbsp;&nbsp;&nbsp;&nbsp;sleep( 1);

&nbsp;&nbsp;&nbsp;&nbsp; //5.&nbsp;关闭连接
&nbsp;&nbsp;&nbsp;&nbsp;close(connectfd);
&nbsp;&nbsp;&nbsp;&nbsp; return&nbsp; 0;
}
编译服务端和客户端的代码: 先启用服务端: 然后用 tcpdump 工具开始抓包,命令如下: tcpdump&nbsp;-i&nbsp;lo&nbsp;tcp&nbsp;and&nbsp;port&nbsp;8888&nbsp;-s0&nbsp;-w&nbsp;/home/tcp_close.pcap
然后启用客户端,可以看到,与服务端连接成功后,发完数据就退出了。 此时,服务端的输出: 接下来,我们来看看抓包的结果。 可以看到,TCP 挥手次数是 3 次。所以,下面这个结论是没问题的。

结论:当被动关闭方(上图的服务端)在 TCP 挥手过程中,「没有数据要发送」并且「开启了 TCP 延迟确认机制(默认会开启)」,那么第二和第三次挥手就会合并传输,这样就出现了三次挥手。

实验二

我们再做一次实验,来看看 关闭 TCP 延迟确认机制,会出现四次挥手吗?客户端代码保持不变,服务端代码需要增加一点东西。在上面服务端代码中,增加了打开了 TCP_QUICKACK (快速应答)机制的代码,如下: 编译好服务端代码后,就开始运行服务端和客户端的代码,同时用 tcpdump 进行抓包。抓包的结果如下,可以看到是四次挥手。

所以,当被动关闭方(上图的服务端)在 TCP 挥手过程中,「 没有数据要发送」,同时「关闭了 TCP 延迟确认机制」,那么就会是四次挥手。

设置 TCP_QUICKACK 的代码,为什么要放在 read 返回 0 之后?

我也是多次实验才发现,在 bind 之前设置 TCP_QUICKACK 是不生效的,只有在 read 返回 0 的时候,设置 TCP_QUICKACK 才会出现四次挥手。网上查了下资料说,设置 TCP_QUICKACK 并不是永久的,所以每次读取数据的时候,如果想要立刻回 ACK,那就得在每次读取数据之后,重新设置 TCP_QUICKACK。而我这里的实验,目的是为了当收到客户端的 FIN 报文(第一次挥手)后,立马回 ACK 报文,所以就在 read 返回 0 的时候,设置 TCP_QUICKACK。当然,实际应用中,没人会在我这个位置设置 TCP_QUICKACK,因为操作系统都通过 TCP 延迟确认机制帮我们把四次挥手优化成了三次挥手了,这本来就是一件好事呀。

总结

当被动关闭方在 TCP 挥手过程中,如果「没有数据要发送」,同时「没有开启 TCP_QUICKACK(默认情况就是没有开启,没有开启 TCP_QUICKACK,等于就是在使用 TCP 延迟确认机制)」,那么第二和第三次挥手就会合并传输,这样就出现了三次挥手。所以,出现三次挥手现象,是因为 TCP 延迟确认机制导致的。完!

TCP 的四次挥手,可以变成三次挥手吗?相关推荐

  1. 计算机网络-TCP的运输连接管理(三次握手,四次挥手)补充一下为什么不能将四次挥手改为三次挥手

    hello,朋友们.今天咱们分享一下TCP连接建立与释放问题(三次握手与四次挥手问题) 1.简单介绍 基础知识了解(仅代表个人简单理解) SYN        同步(一个信号   代表自己的状态) F ...

  2. 为什么是四次挥手不是三次挥手

    1. 四次挥手过程以及状态 报文段1:主动关闭方向被动关闭方发送结束报文段. 报文段2:被动关闭方发送结束报文段确认,此时主动关闭方数据已经传输完毕. 报文段3:被动关闭方数据也发送完毕,发送结束报文 ...

  3. 【计算机网络】传输层 : TCP 连接管理 ( TCP 连接建立 | 三次握手 | TCP 连接释放 | 四次挥手 )

    文章目录 一.TCP 连接管理 二.TCP 连接建立 三.TCP 连接建立 相关报文段 字段 四.SYN 洪泛攻击 五.TCP 连接释放 一.TCP 连接管理 TCP 传输数据过程 : 建立连接 -& ...

  4. TCP 四次挥手,可以变成三次挥手吗?

    作者:小林coding 计算机八股文网站:https://xiaolincoding.com 大家好,我是小林. 虽然我们在学习 TCP 挥手时,学到的是需要四次来完成 TCP 挥手,但是在一些情况下 ...

  5. TCP的连接和释放连接(三次握手和四次挥手的过程)

    参考文章: javascript - 看图理解TCP的三次握手和四次挥手_个人文章 - SegmentFault 思否 TCP'三次握手'和'四次挥手'(通俗易懂)_大黄的Java笔记的博客-CSDN ...

  6. 【漫画】TCP断开连接为什么是四次挥手,不是二次挥手/三次挥手?

    前情回顾:[漫画]TCP连接为什么是三次握手,而不是两次握手,也不是四次握手? 乔戈里和小萌一起去美食城吃了午饭 小萌:额...哦!这就是两次挥手,我这里就好比是服务端还有消息没发送完,乔哥你的客户端 ...

  7. TCP四次挥手不同情况的研究(正常状态,三次挥手,以及CLOSING和RST)

    文章目录 TCP四次挥手不同情况的研究 1.新手加油站 1.1 TCP四次握手流程图 1.2 TCP头部 1.3 四次握手流程详解 2.针对挥手不同情况的实验和研究 2.1 正常情况 2.2 三次挥手 ...

  8. TCP三次挥手四次握手(面试总结)

    1. 为什么建立连接协议是三次握手,而关闭连接却是四次握手呢? 全双工通信. 这是因为服务端的LISTEN状态下的SOCKET当收到SYN报文的建连请求后,它可以把ACK和SYN(ACK起应答作用,而 ...

  9. 三次握手和四次挥手图解_三次握手和四次挥手简单理解

    TCP三次握手 TCP标志位 TCP在其协议头中使用大量的标志位或者说1位(bit)布尔域来控制连接状态,一个包中有可以设置多个标志位. TCP是主机对主机层的传输控制协议,提供可靠的连接服务,采用三 ...

  10. Python面试总结(四)ip正则与三次握手四次挥手

    1.请简述值传递和引用传递的区别? 值传递仅仅传递的是值 引用传递,传递的是内存地址,修改后会改变内存地址对应储存的值. 2.请手写一个匹配ip的正则表达式? 第一种方法: ((2[0-4]\d|25 ...

最新文章

  1. 自动驾驶有量子飞跃式改进,马斯克称年内实现L5级别自动驾驶?
  2. 分享下我的博客园CSS
  3. bazel 链接第三方动态库_Linux 动态库与静态库制作及使用详解
  4. iOS 远程通知(Remote Notification)和本地通知(Local Notification)
  5. eeprom stm8l 擦除 读写_[STM8L]EEPROM操作读与写
  6. delphi 发送html邮件,delphi发送html带附件邮件
  7. Leetcode--85. 最大矩形
  8. 系统架构设计师 - 信息安全技术
  9. boost any 实现万能容器_全面剖析 C++ Boost 智能指针!| CSDN 博文精选
  10. 一文读懂-Impala
  11. 跨境电子商务独立站如何找到热门的利基市场
  12. .net String Formatter 格式转换
  13. 造成跨域的原因和解决方法
  14. 电机编码器调零步骤_什么是无刷直流电机换向的最有效方法?
  15. 用iptables做软路由实现共享上网
  16. Spark Architecture
  17. C#积木游戏(改编自DevExpress GridTetris)
  18. MySQL事务之可重复读
  19. adb命令 关机与重启
  20. ie和chrome浏览器下onproperty事件oninput onpropertychange的相应和相应属性的获取

热门文章

  1. 【服务器管理】Ubuntu的一次惊心动魄的查杀挖矿病毒的经历:病毒伪装成python
  2. 拼多多 标题 html,【拼多多如何变成新用户】拼多多老用户变新用户教程_拼多多砍价网...
  3. delphi去掉字段前后的引号_delphi 单引号在字符串中使用方法
  4. 山峰和山谷 Ridges and Valleys
  5. 狂神说笔记——JavaScript快速入门11
  6. 数据库考试内容(MYSQL)
  7. 关于国际论文中,国内外人名顺序的问题
  8. 有关Nodejs的视频教程
  9. 客户关系管理 期末复习
  10. PC微信机器人之实战分析通过wxid获取用户信息