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. 【Curl】【转】curl用法!

    curl基础用法! www.ruanyifeng.com/blog/2019/09/curl-reference.html

  2. day03 Pyhton学习

    昨日回顾 1.while循环 语法 while 条件: 语句 else: 语句 执行语句:判断语句是否为真.如果真,执行循环,然后再次判断条件,如果不满足执行else语句. break 结束循环 co ...

  3. day09 Pyhton学习

    一.昨日内容回顾 文件操作 open(文件路径,mode="模式",encoding="编码") 文件路径: 1.绝对路径 从磁盘根目录寻找 2.相对路径 相对 ...

  4. 【自学编程】新手经常遇到的10大C语言基础算法,珍藏版源码值得收藏!

    算法是一个程序和软件的灵魂,作为一名优秀的程序员,只有对一些基础的算法有着全面的掌握,才会在设计程序和编写代码的过程中显得得心应手.本文是近百个C语言算法系列的第二篇,包括了经典的Fibonacci数 ...

  5. php使用xpath爬取内容

    <?php $html = file_get_contents('https://tieba.baidu.com/f?kw=%C9%EE%BB%A7&fr=ala0&loc=re ...

  6. scrapy-下载器中间件 随机切换user_agent

    from faker import Faker class MySpiderMiddleware(object): def __init__(self): self.fake = Faker() de ...

  7. phpstorm 使用xdebug

    一.在phpstudy配置 开启xdebug的zend扩展,在php.ini 中添加下面的代码: [xdebug] zend_extension = "D:\phpstudy_pro\Ext ...

  8. is, ==, id 用法、代码块和缓存机制

    id(): 获取对象的内存地址:print(id(i)) == : 比较两边的值是否相同 is : 判断内存地址是否相同 id相同,值一定相同 值相同,id不一定相同 代码块: Python是由代码块 ...

  9. docker启动redis并设置密码

    docker启动redis并设置密码: docker run -d --name redis -p 6379:6379 redis --requirepass "password" ...

  10. 正式班D14

    2020.10.23星期五 正式班D14 9.5 文件处理三剑客(支持|) 9.5.1 sed流式编辑器 事先制定好编辑文件的指令,让sed自动完成对文件的整体编辑(同一时间内存中只有文件中一条) # ...