TCP序列号与ack的计算(Wireshark抓包分析)

前言

最近在做一个关于TCP的实验,需要了解TCP的seq和ack的发送机制,看了很多文章之后,再结合着实际的测试,归纳出了seq和ack的计算方法
这里不得不说一句,关于怎么计算seq和ack网上的教程和文章实在是太少了,一搜TCP,出来的就是三次握手和四次挥手,难道TCP只需要握手和挥手就行了吗。

结论

先说结论:
在已经建立好连接的TCP上(只考虑数据包和ack包),seq和ack的计算规则为
本次要发送的包的 seq = 上一个发送的包的seq + 上一个发送的包的长度(不含包头)
本次要发送的包的 ack = 上一个接收到的包的seq + 上一个接收到的包的长度(不含包头)

分析

按照我的理解,seq和ack都是指针 。seq指示要发送的包在窗口中的起始位置 ,ack指示已经接收的包的位置
下面我就用我wireshark抓的包和画的实例图一起分析一下

第一个包:

在这里插入图片描述
在这里插入图片描述
第一个包是 客户端 发向 服务器 的数据包,长度为 517 字节
首先,seq和ack的值都是1,这是握手后的状态。
然后,第一个包对应TCP流行图的第四行
此时要发送的数据包的起始位置是1,还没有已经接收到的数据包,所以ack 也为 1。

第二个包

在这里插入图片描述
在这里插入图片描述
再第一个包发完之后,客户端的seq(下一个要发送数据的其实位置)要后移 517 字节
第二个包是 服务器 发向 客户端 的ack包
服务器的接收窗口收到数据,所以ack = 已收到的字节 = 1 + 517 = 518 = 上一个收到的包的seq + 包长
由于服务器并没有发送数据,所以seq(下一个要发送数据起始位置)保持不变
客户端接收到了ack包,但是这个包没有数据,所以客户端的接收窗口不变。

第三个包

在这里插入图片描述
在这里插入图片描述
第三个包是 服务器 发往 客户端 的数据包 长度为 96
seq(发送数据的起始位置)= 1;
ack (接收数据的末尾)= 518;

第四个包

在这里插入图片描述
在这里插入图片描述

在服务器发送第三个包之后, seq后移 96 个字节,变成 96 + 1 =97
客户端收到第三个包,ack后移96,变成 96 + 1 =97
第四个包是 服务器 发往 客户端 的 数据包长度 6 个字节
所以这个包的 seq(发送数据的起始位置) = 97
由于服务器没有接收到新的数据,所以 ack(接收数据的末尾) = 518(不变)

第五个包

在这里插入图片描述
在这里插入图片描述

服务器发送第四个包后,seq后移6字节(图中蓝色部分),seq = 97 + 6 = 103
客户端收到第四个包,ack后移6字节 ack = 97 + 6 = 103
第五个包是 服务器发送 给 客户端 的数据包(长度45字节)
容易计算 seq = 103
ack = 518(不变)

第六个包

在这里插入图片描述
在这里插入图片描述
这是一个由客户端发往服务器的ack包,但是需要注意的是,这个包是对 第四个包(长度为6)的数据包的ack
seq(发送数据的起始位置) = 518
ack (已经接收的包)= 1 + 96 + 6 = 103

以此类推

以此类推,就可以分析每一个数据包的seq和ack的值了,附上完整的流行图
在这里插入图片描述

结语

我是用指针的方式来理解seq和ack的,经过学习才发现,理论上(自顶向下)的TCP的和实际的TCP还是有非常大的差别的。
这篇文章是我自己的总结,难免有不对的地方,希望大家指正。

  • 12
    点赞
  • 41
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
计算TCP头长度时,需要考虑TCP头的各个字段。TCP头的长度是由数据偏移字段决定的,数据偏移字段占TCP头的第12个字节,它指定了TCP头的长度以及TCP头后面跟随的数据的起始位置。数据偏移字段的值为4位,表示TCP头中32位字(4个字节)的数量。 因此,计算TCP头的长度需要进行以下步骤: 1. 获取TCP头的数据偏移字段值,即取TCP头的第12个字节,将其右移4位。 2. 将获取到的数据偏移字段值乘以4,得到TCP头的长度。 以下是C语言代码示例: ```c #include <stdio.h> #include <stdint.h> // TCP头结构体 struct tcp_header { uint16_t src_port; // 源端口号 uint16_t dest_port; // 目标端口号 uint32_t seq_num; // 序列号 uint32_t ack_num; // 确认号 uint8_t data_offset; // 数据偏移 uint8_t flags; // 标志位 uint16_t window_size; // 窗口大小 uint16_t checksum; // 校验和 uint16_t urgent_ptr; // 紧急指针 }; // 计算TCP头长度 int calc_tcp_header_len(struct tcp_header *tcp_hdr) { return (tcp_hdr->data_offset >> 4) * 4; } int main() { // 构造一个TCP头 struct tcp_header tcp_hdr = { .src_port = 1234, .dest_port = 5678, .seq_num = 0, .ack_num = 0, .data_offset = 0x50, // 数据偏移为5,即20个字节 .flags = 0, .window_size = 8192, .checksum = 0, .urgent_ptr = 0 }; // 计算TCP头长度 int tcp_hdr_len = calc_tcp_header_len(&tcp_hdr); printf("TCP header length: %d bytes\n", tcp_hdr_len); return 0; } ``` 输出结果: ``` TCP header length: 20 bytes ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值