1.握手

说明:

  下面涉及 FIN,SYN,ACK之类数据时,都是由TCP服务收发,

  涉及 accept, listen 之类api,都是 应用进程 完成。

  都统一使用 客户端,服务端描述,请自行分辨。

(1)首先描述下3次握手,TCP协议做了什么。

客户端,主动打开,发送自己的序列号SYNj,并期待对方回复ACKj+1

服务端,被动打开,接送自己的序列号SYNk和ACKj+1,并期待对方回复ACKk+1

客户端,接收对方ACK,己方打开完成,接收对方SYN,发送ACKk+1

服务端,接收对方ACK,己方打开完成。

分析下:

  首先打开分为,被动打开和主动打开。

  当接收到对方的ACK,则己方打开完成,则可以使用socket进行读写操作,但并不保证正常。

(2)结合 系统调用分析三次握手

客户端,listen,进行被动打开

服务端,connect ,进行主动打开,发送SYNj(如果乙方此时没有listen 完成,则connect 失败)

客户端,回复SYNk + ACKj+1, 应用层无任何变化。

服务端,收到 ACKj+1 + SYNk,并回复 ACKk+1, 甲方打开完成,connect 返回。

客户端,收到ACKj+1,可以进行 read,write。

服务端,accept,如果乙方没有收到 SYNj 则会阻塞,否则返回

服务端,收到ACKk+1,可以进行 read, write

总结下:

  从应用层角度,三次握手的开始,

    主动方,始于 connect,结束于 connect

    被动方,始于 listen ,结束于 listen

  accept 是被动方处理已经完成握手的主动方。

(3)实验

client

if  (0 > (connect(fd, (struct sockaddr *)&addr, sizeof(addr)))) {
perror("connect");
exit(-1);
} printf("success to connect\n"); write(fd, "hello world", strlen("hello world")); printf("write over\n"); close(fd);

server

        if (0 > listen(fd, 10)) {
perror("listen");
exit(-1);
} #if 1
printf("sleep for delay accept...\n");
sleep(5);
printf("sleep over\n");
#endif if (0 > (c_fd = accept(fd, (struct sockaddr *)&c_addr, &len))) {
perror("accept");
exit(-1);
} printf("success to accept\n"); while ((nbytes = read(c_fd, msg, MAX_READ))) {
msg[nbytes] = 0;
printf("%s", msg);
}
printf("\n"); if (0 > (nbytes = write(c_fd, "bye", 3))) {
perror("write");
exit(-1);
} printf("i have see bye\n"); close(fd);
close(c_fd);

实验结果

sleep for delay accept...
sleep over // 这里客户端已经通信结束,程序退出
success to accept
hello world
i have see bye

可以发现,客户端虽然已经挥手,但是服务端仍然视为握手完成,并正常的进行通信。

用netstat查看

可以发现,客户端进程虽然死掉,但打开的TCP服务并没有关闭,他在等待四次挥手完成。

因此服务端,虽然延迟处理,却依旧能正常处理通信。、

2.四次挥手

客户端,主动关闭,close,发送 FINm

服务端,被动关闭,接收FINm,发送ACKm+1

一段时间后,服务端,调用close,发送FINn

客户端,接收FINn,发送ACKn+1

总结下:

  当服务端接收到FIN时,注意,所有的TCP数据都由TCP服务处理,服务端的TCP收到FIN,并向服务端进程发送文件结束符EOF。

  EOF按照顺序在通信数据后面。

  当服务端进程read获得EOF,适时主动调用close,以发送 FINn。

  

  在客户端close,服务端没有close的阶段,称为 半关闭。

  这时,服务端进程可以发送数据,数据会交给客户端TCP服务,但不会上交给应用层。所以客户端进程是否存在,没关系。

UNP——第二章,TCP握手与挥手分析的更多相关文章

  1. TCP 握手和挥手图解(有限状态机)

    1.引言 TCP 这段看过好几遍,老是记不住,没办法找工作涉及到网络编程这块,各种问 TCP .今天好好整理一下握手和挥手过程.献给跟我一样忙碌,找工作的童鞋,欢迎大神批评指正. 2.TCP 的连接建 ...

  2. 第二章 TCP/IP 基础知识

    第二章 TCP/IP 基础知识   TCP/IP  transmission control protocol and ip internet protocol 是互联网众多通信协议中最为著名的.   ...

  3. 抓包分析 TCP 握手和挥手

    前言 首先需要明确的是 TCP 是一个可靠传输协议,它的所有特点最终都是为了这个可靠传输服务.在网上看到过很多文章讲 TCP 连接的三次握手和断开连接的四次挥手,但是都太过于理论,看完感觉总是似懂非懂 ...

  4. UNP——第二章,端口号,套接字对,TCP,UDP输出

    1.端口号 端口号用于区分使用相同协议的进程. TCP69 与 UDP69 是不同的. 端口号范围 0 - 65535, 其中 0- 1023 是保留端口. 2.套接字对 TCP服务通过套接字对,唯一 ...

  5. UNP——第二章,TCP状态,TIME_WAIT

    状态可以用 netstat 验证 加粗线为 数据交换. 可以看出,TCP在 建立连接和 关闭连接,耗费资源, 因为UDP只需要两次数据通信即可. 但UDP没有可靠传输,和流量控制. 上面协商的MSS为 ...

  6. CCNA第二章TCP/IP简介考试要点学习笔记

    1.描述网络是如何工作的     DoD过程/应用层 -- OSI应用.表示和会话层(定义了结点到结点的应用通信协议以及对用户界面规范的控制): DoD主机到主机层 -- OSI传输层(保证了数据包的 ...

  7. UNP——第二章,常见协议概述

    1.为什么要了解协议 程序员与协议合作,完成应用. 了解协议是为了了解协议完成了什么,提供了什么服务,自己还应该做什么. 2.从协议的角度,套接字是什么 套接字是协议的接口, IP套接字,代表可使用I ...

  8. [Maven实战-许晓斌]-[第二章]-2.3安装目录分析

    bin boot conf settings.xml非常重要 这个是maven安装包自带的settings.xml 通常我们会放在习惯路径,C:\Users\admin\.m2\下面 即  用户路径\ ...

  9. TCP/IP三次握手四次挥手分析

    流程图 全部11种状态 客户端独有的:(1)SYN_SENT (2)FIN_WAIT1 (3)FIN_WAIT2 (4)CLOSING (5)TIME_WAIT 服务器独有的:(1)LISTEN (2 ...

随机推荐

  1. ASP.Net Core3.1 生成二维码填坑

    ASP.Net Core3.1 使用QrCode生成二维码 部署到Linux报错 The type initializer for 'System.DrawingCore.GDIPlus' threw ...

  2. day27 Pyhton 面向对象02 组合

    # 组合 # 什么是组合 : 一个类对象的属性是另外一个类的对象 class Person: def __init__(self,name,sex,hp,mp,ad): self.name = nam ...

  3. [leetcode] 剑指 Offer 专题(一)

    又开了一个笔记专题的坑,未来一两周希望能把<剑指Offer>的题目刷完

  4. 【C语言/C++程序员编程】一小时做出来的数字雨(一颗开花的树)!

    相信大家看过许许多多的关于计算机黑客.骇客.人工智能.AI方面的电影,每当黑客入侵某个五角大楼,某个网站时,都会出现这样一副画面: 入侵 或者这样的: 数字雨 然后就轻而易举的成功入侵夺取管理员权限了 ...

  5. docker系统化学习图文+视频教程

    1.背景 博客对应的视频课程: 9.9元在线学习:https://study.163.com/course/courseMain.htm?share=2&shareId=40000000033 ...

  6. spring boot:用zxing生成二维码,支持logo(spring boot 2.3.2)

    一,zxing是什么? 1,zxing的用途 如果我们做二维码的生成和扫描,通常会用到zxing这个库, ZXing是一个开源的,用Java实现的多种格式的1D/2D条码图像处理库. zxing还可以 ...

  7. 五分钟详解MySQL并发控制及事务原理

    在如今互联网业务中使用范围最广的数据库无疑还是关系型数据库MySQL,之所以用"还是"这个词,是因为最近几年国内数据库领域也取得了一些长足进步,例如以TIDB.OceanBase等 ...

  8. Python字典的初识、增删改查及嵌套

    为什么要有字典? 列表可以存储大量的数据,但数据间的关联型不强 列表的查询速度相对慢 dict:字典,容器型数据类型 数据类型的分类: 可变与不可变 可变(不可哈希)的数据类型: 列表list,字典d ...

  9. QT/C++插件式框架、利用智能指针管理内存空间的实现、动态加载动态库文件

    QT.C++插件式框架.主要原理还是 动态库的动态加载. dlopen()函数.下面为动态加载拿到Plugininstance对应指针.void**pp=(void**)dlsym(handle,&q ...

  10. MySQL开启日志记录执行过的SQL语句

    当需要分析执行过的SQL语句来判断问题,可以通过打开查询日志功能,但是重启MySQL服务后需要重新配置. 查询日志查询功能: SHOW VARIABLES LIKE 'general%'; gener ...