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. bootStrap小结3

    <!DOCTYPE html> <html lang="en"> <head> <meta http-equiv="Conten ...

  2. hugo官方相关文档地址

    +++ date="2020-10-17" title="hugo官方相关文档地址" tags=["hugo"] categories=[& ...

  3. 如何将vscode代码快速同步到github/gitee上

    用git实现源代码管理几乎是程序员的必备操作,下面是简单实现流程: 在vscode打开代码所在文件夹 在左侧栏点击源代码管理 初始化存储库 添加远程存储库 输入远程仓库地址(没有仓库的要先建个仓) 输 ...

  4. Linux文件元数据和节点表结构

    文件元数据 一块硬盘的分区可以认为有两部分组成,保存元数据的成为节点表,用来保存属性等. 元数据中有个小指针,指向数据存放的实际空间. 元数据(Metadata) 又称中介数据.中继数据,为描述数据的 ...

  5. Linux入门到放弃之六《磁盘和文件系统管理二》

    上一篇博客写到了如何创建卷组和创建逻辑卷,但是有一个问题,需要更大逻辑卷空间怎么办呢? 要求:使用lvextend命令为逻辑卷 mail扩充容量,从卷组 mail_store 上再 划出5GB给逻辑卷 ...

  6. plsql查询中文乱码

    1.查看数据库字符集 select userenv('language') from dual 查看数据库字符集 2.在环境变量中添加并设置变量 变量名:NLS_LANG: 变量值:第一步查询的数据库 ...

  7. 556. 下一个更大元素 III

    556. 下一个更大元素 III 给定一个32位正整数 n,你需要找到最小的32位整数,其与 n 中存在的位数完全相同,并且其值大于n.如果不存在这样的32位整数,则返回-1. 示例 1: 输入: 1 ...

  8. centos7安装oracle版本的jdk

    Hadoop机器上的JDK,最好是Oracle的Java JDK,不然会有一些问题,比如可能没有JPS命令. 如果安装了其他版本的JDK,卸载掉!!! 1,查看是否已经安装了jdk java -ver ...

  9. 将字符串反转的 Java 方法

    Java中经常会用到将字符串进行反转的时候,程序员孔乙己总结了7种反转方法,如下: //方法1 递归方法 public static String reverse1(String s) { int l ...

  10. 转 js调用提交表单。

    今天做网银支付的时候,需要做到点击支付的时候提交订单,然后新窗口打开支付界面. 思路1:window.open(''),这个直接被pass了,因为银行的服务一般都是需要post数据的.就算是可以用ge ...