Wireshark 网络包分析实战五:seq/ack在TCP通信不同阶段的数值关系

Overview

  • 当我们说,TCP是可靠的,我们在表达什么? 或者说,TCP协议是怎么保证可靠的?

实现机制其实非常简单,就是靠ackseq机制,我们直接看TCP Header

也就是Sequence NumberAcknowledgment NumberTCP的通信与HTTP不通,它并不是一个请求对应一个响应,为了加快传输的效率(因为一来一回就是一个RTT啊),TCP的通信机制允许一次发送多个数据包,然后多个发送包可以对应一个响应包,也就是所谓ACK包。

TCP到底如何保证发送的数据被接收到了呢?就是靠接收方返回的数据包中的ack字段。

比如ack=1000,接收方告诉发送方:嗨,老兄。你刚才发了很多数据嘛,我跟你讲,1000字节以前的数据我都收到了,下次发送就从1000字节开始发哈。

seqack就是这么重要。下面让我们学习一下在TCP通信的不同阶段,它们之间可能都有什么数量关系吧~

Example

Three-way Handshake

在三次握手和四次挥手的过程中,没有应用层数据,即有效载荷为0.

但是三次握手中的SYN标志、四次挥手的FIN标志都算作:载荷为1,即实际上Len=1

眼见为实,以三次握手为例:

我们看30号包:seq=0,载荷TCP Len 也为0,但31号包是怎么说的?

31号包的ACK=1,好比30号包发送了1 byte的数据。

30号包真的没有携带

1、TCP Len 显式为0

2、IP包显式长度为60Length = IP Header + TCP Header + TCP Len,其中TCP Header包含固定部分以及Options部分,固定部分为20字节,Options部分为20字节,IP Header部分为20字节,加在一起,可以得出TCP Len = 0

32号包的seq=1,算是坐实了SYN这个标志值1 byte

clientSYN标志占据1 byte,同样的,server端的SYN标志也占用1 byte

何以见得?还是看32号包:

32号包的ACK=1

关于这一点,Stevens在《TCP/IP Illustrated,Volume 1》是这么说的:

The sequence number of the first byte of data sent on this direction of the connection is the ISN plus 1 because the SYN bit field consumes one sequence number. As we shall see later, consuming a sequence number also implies reliable delivery using retransmission. Thus, SYNs and application bytes (and FINs, which we will see later) are reliably delivered. ACKs, which do not consume sequence numbers, are not.

让我们直接来看真实的网络包吧~

Data Transmission

正常通信的过程,那就是对端的Ack=发送端的Seq+Len

这是一次简单的HTTP GET 请求。

我们看33号包,client发出GET 请求,其seq=1,载荷TCP Len=106,它下一次数据包的seq应该就是从seq+tcp len = 107开始。

何以见得?请看34号包,server表示收到了client,也就是33号包发过来的请求,它进行了确认ack=107。正如开头所说,这就是TCP可靠性的真谛。

紧接着,36号包,client再次发送了数据包,它的seq果真就是107bingo

TCP Keep-alive

发送端seq龟缩大法

仔细观察,TCP保活机制中的包会有一种seq回退现象。

  • 25 号包

seq=1539,nextSeq=1578,说明它下一个包的seq应该从1578开始发起。

但它并没有!

  • 27 号包 - keep-alive

它的seq居然是从1577开始,比1578少了一个byte,这是为什么?

向后看,原来27号包是TCP Keep-Alive包,seq会发生回退。

  • 28号包 - keep-alive ack

作为27号包的响应包,它的ack=1578。如果后面是正常的数据传输,发送端的seq确实应该seq=1578开始传输。但如果发送端依然是保活包,也就是keep-alive的话,seq依然会从1578-1=1577开始。你不信的话,就看29号包!

Application Keep-alive

本质上就是应用层进行了数据传输。

上述网络包表示应用层自己实现的保活机制。保活包中的请求和响应都分别携带1 byte的数据,这一点可以从图中的TCP Payload Length看出。

所以相比TCP自身的保活机制,这里没什么稀奇古怪的。

250号包来说,seq=6190len=1,那么确认250号包的ack应该为seq+len=6191

查看251号包中的ack字段,发现的确如此!

这对于TCP来说,其实就是普普通通的数据传输过程。

Summary

让我们总结一下:

  • 三次握手、四次挥手:ack = seq+11指的是SYNFIN标志,表示它们很重要)
  • 数据传输:ack = seq + len
  • TCP心跳保活:keep-alive包的seq会发生回退减1的情况,keep-alive-ackack是正常的。
  • 应用层的心跳保活:ack = seq + len,只不过是一种特殊的数据传输过程而已啦

Reference

  • 极客时间《网络排查案例课》
  • 林沛满《Wireshark网络分析的艺术》
  • Richard Stevens《TCP/IP Illustrated, Volume 1》
相关推荐
友友马15 分钟前
『 Linux 』网络层 - IP协议(一)
linux·网络·tcp/ip
hgdlip5 小时前
主IP地址与从IP地址:深入解析与应用探讨
网络·网络协议·tcp/ip
今天我刷leetcode了吗5 小时前
docker 配置同宿主机共同网段的IP 同时通过通网段的另一个电脑实现远程连接docker
tcp/ip·docker·电脑
lwprain6 小时前
安装支持ssl的harbor 2.1.4 docker 19.03.8 docker-compose 1.24.0
网络协议·ssl·harbor
软件技术员6 小时前
Let‘s Encrypt SSL证书:acmessl.cn申请免费3个月证书
服务器·网络协议·ssl
黑客呀8 小时前
抓包 127.0.0.1 (loopback) 使用 tcpdump+wireshark
测试工具·wireshark·tcpdump
爱分享的码瑞哥8 小时前
Python爬虫中的IP封禁问题及其解决方案
爬虫·python·tcp/ip
_不会dp不改名_8 小时前
HCIA笔记3--TCP-UDP-交换机工作原理
笔记·tcp/ip·udp
co0t8 小时前
计算机网络(14)ip地址超详解
服务器·tcp/ip·计算机网络
C++忠实粉丝8 小时前
计算机网络socket编程(3)_UDP网络编程实现简单聊天室
linux·网络·c++·网络协议·计算机网络·udp