【Linux C | 网络编程】详细介绍 “三次握手(建立连接)、四次挥手(终止连接)、TCP状态”

😁博客主页😁:🚀https://blog.csdn.net/wkd_007🚀
🤑博客内容🤑:🍭嵌入式开发、Linux、C语言、C++、数据结构、音视频🍭
🤣本文内容🤣:🍭介绍“三次握手(建立连接)、四次挥手(终止连接)、TCP状态” 🍭
😎金句分享😎:🍭你不能选择最好的,但最好的会来选择你——泰戈尔🍭

本文未经允许,不得转发!!!


在这里插入图片描述

🎄一、概述

一个TCP连接通常分为3个阶:建立连接、数据传输(也称作“连接已建立”)、终止连接。其中建立连接、终止连接就涉及到经常听到的三次握手,四次挥手。并且,三次握手,四次挥手 常常出现在面试中,大部分应试者一上来就是知道回复:

首先,客户端向服务端发送一个"SYN"分节;
然后,服务端向客户端发送一个"SYN+ACK"分节
最后,客户端向服务端发送一个ACK分节表示确认

这样回答是生硬的、欠缺的、不够深度的。

那请问:
1、三次握手的目的是什么?
2、为什么需要三次握手?
3、两次握手行不行?
4、SYN报文可以携带数据吗?
5、三次握手过程中,每次发送报文后,TCP处于什么状态?
6、四次挥手的过程是怎样的?
7、为什么握手是三次,挥手却要四次?
8、什么是半关闭状态?

如果上面几个问题都懂了,那不必往下看了。


在这里插入图片描述

🎄二、三次握手(建立连接)

TCP协议一个面向连接的协议,那么这个连接是怎么建立的呢?

TCP的连接就是通过所谓的三次握手来建立的,在网络编程中,TCP连接的建立是发生在客户端的connect函数和服务端的accept函数中的。所以,我们常常可以看到使用套接字API写TCP客户端时,都是依次调用socket、connect,而服务器常见的是依次调用socket、bind、listen、accept

✨2.1 三次握手的过程

建立一个TCP连接会进行下面步骤:

  • 1、TCP客户端调用socket函数获取到套接字句柄之后,调用connect函数发起主动打开(active open),这将导致客户端TCP发送一个SYN报文段(一个TCP头部的SYN位字段置位的TCP/IP数据包),并指明自己想要连接的端口号和它的客户端初始序列号(Initial Sequence Number, ISN)。通常,客户端还会借此发送一个或多个选项。
  • 2、TCP服务端在调用依次socket、bind、listen之后,执行被动打开(passive open)。服务端接收到客户端的SYN报文后,也发送自已的SYN报文段作为响应,并包含了它的初始序列号;此外,为了确认客户端的SYN,服务端将客户端的初始序列号(Initial Sequence Number, ISN)数值加1后作为返回的ACK数值。
  • 3、客户端为了确认服务器的SYN,将服务端的初始序列号(Initial Sequence Number, ISN)数值加1后作为返回的ACK数值。

在这里插入图片描述

经过上面三个步骤,TCP连接就建立成功了,之后可以通过read、write等函数交互数据了。

注意:三次握手是在传输层完成的,主要通过操作系统内核的网络协议栈来处理。connect仅仅是通知给linux内核,让linux内核自动完成TCP三次握手连接。
上面三路握手过程图中,看到客户端的SYN报文到达是,accept函数已经调用,容易错误地认为需要accept函数调用后,才可以建立三次握手。其实,只要服务端调用listen函数后就可以正常建立三次握手了,accept调不调用都不影响三次握手的建立,accept函数的作用是从服务端的已连接队列中获取一个已建立的连接,所以如果服务端需要与客户端通信就需要调用accept。


✨2.2 为什么需要三次握手

三次握手的目的不仅在于让通信双方了解一个连接正在建立,还在于利用数据包的选项来承载特殊的信息,交换初始序列号(Initial Sequence Number, ISN)。

另外,必须是三次握手后,才可以让客户端、服务端都确认对方的发送、接收能力是否正常:

  • 第一次握手(cli -> ser):
    客户端检测服务端的接收能力。
    服务端收到客户端的SYN后,确认了客户端具有发送能力。
  • 第二次握手(ser -> cli):
    客户端收到服务端的SYN+ACK,确认了服务端具有发送能力,同时说明服务端已经收到自己的SYN(具有接收能力);
    服务端检测客户端的接收能力。
  • 第三次握手(ser -> cli):
    服务端收到客户端的ACK,说明客户端已经收到自己的SYN(具有接收能力)。

如果只有两次握手行不行呢?
只有两次握手的话,服务端消息发出去后,没受到响应,无法确认客户端是否可以接收信息。所以需要三次握手才可以建立连接。


注意:TCP的SYN段也能够承载应用数据。由于伯克利的套接字API不支持这种方式,因此它也很少为人所用

在这里插入图片描述

🎄三、四次挥手(终止连接)

TCP连接建立后,客户端、服务端会交互数据,最后连接的任一方可以发起终止,一般是客户端发起终止,但有些服务端处理完客户端的请求后也会发起关闭操作。

✨2.1 四次挥手的过程

  • 1、TCP连接的任一端首先调用closeshutdown函数,执行主动关闭,这会向TCP连接对端发送一个FIN报文段,并指明当前的序列号,该FIN段还包含了一个ACK段用于确认对方最近一次发来的数据。
  • 2、接收到FIN报文段的一端执行被动关闭,将发过来的序列号加1作为响应的ACK数值。上层的应用程序会接收到一个文件结束符(end-of-file)表示该连接不会再发送数据过来,read函数会返回0;
  • 3、一段时间后,接收到这个文件结束符的应用程序也会调用closeshutdown函数关闭它的套接字,这会导致它的TCP也发送一个FIN报文段
  • 4、接收到这个最后FIN报文段的一端会将接收到的序列号加1作为响应的ACK数值。

在这里插入图片描述

✨2.2 为什么需要四次挥手

为什么连接的时候是三次握手,终止的时候却是四次握手?

  • 三次握手:因为服务端接收到SYN报文后,可以将确认的ACK数组添加到自己的SYN报文中一并发出。
  • 四次挥手:因为TCP存在半关闭状态,就是允许连接的一端在结束它的发送后还能接收来自另一端数据。一般情况下,收到FIN报文后,并不会立刻关闭socket,而是先回复一个ACK报文,因为此时可能还有一些数据正在发送,等所有数据发送后,才会发送FIN报文,所以需要四次握手。但这并不是必然的,在《Unix网络编程卷1》有提到,第2、3步骤都出自执行被动关闭的那一端,可能被合成一个报文。

在这里插入图片描述

🎄四、TCP状态

TCP状态共11种:

  • 1、 客户端独有的:(1)SYN_SENT (2)FIN_WAIT1 (3)FIN_WAIT2 (4)CLOSING (5)TIME_WAIT 。
  • 2、服务端独有的:(1)LISTEN (2)SYN_RCVD (3)CLOSE_WAIT (4)LAST_ACK 。
  • 3、共有的:(1)CLOSED (2)ESTABLISHED 。
LISTEN       - 监听来自客户端的连接请求; 
SYN_SENT     - 在发送连接请求后等待响应的状态; 
SYN_RCVD 	 - 在收到和发送一个连接请求后等待对端确认的状态; 
ESTABLISHED  - 代表一个已建立的连接,数据开始交互; 
FIN_WAIT1    - 主动关闭,发送完FIN,等待对端确认;
FIN_WAIT2    - 主动关闭,发送完FIN且已收到对端确认,等待接收FIN; 
CLOSE_WAIT   - 被动关闭,接收完FIN且已确认,等待发送FIN; 
CLOSING      - 发送完FIN后,还没收到ACK,就收到FIN并确认,说明对端也正在关闭,等待对端确认; 
LAST_ACK     - 被动关闭,发送完FIN,等待对端确认;
TIME-WAIT    - 等待2MSL,以确保对端TCP接收到连接终止的确认; 
CLOSED       - 没有任何连接状态;

在这里插入图片描述

在这里插入图片描述

🎄五、总结

👉本文详细介绍了三次握手(建立连接)的过程、目的,以及四次挥手的过程、目的,最后讲解了TCP连接的11种状态。

如果真的面试遇到了三次握手、四次挥手的问题,建议这么做,
1、先把过程图画出来;
2、把三次握手、四次挥手的过程展开说一下,然后看情况,可以直接结束了,也可以再说说为什么需要三次握手、四次挥手;
3、最后,把最前面的几个问题都准备好,以防提问到。

在这里插入图片描述
如果文章有帮助的话,点赞👍、收藏⭐,支持一波,谢谢 😁😁😁

参考资料:
《Unix 网络编程卷1》
《TCP/IP 详解卷1》
《我终于搞懂了TCP的三次握手和四次挥手》https://blog.csdn.net/weixin_45393094/article/details/104965561
《面试官,不要再问我三次握手和四次挥手》https://blog.csdn.net/hyg0811/article/details/102366854

最近更新

  1. TCP协议是安全的吗?

    2024-02-01 14:28:01       16 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-02-01 14:28:01       16 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-02-01 14:28:01       15 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-02-01 14:28:01       18 阅读

热门阅读

  1. 【SparkML系列2】DataSource读取图片数据

    2024-02-01 14:28:01       28 阅读
  2. 网络安全笔记

    2024-02-01 14:28:01       30 阅读
  3. 软件工程知识梳理2-需求分析

    2024-02-01 14:28:01       33 阅读
  4. 开发安全之:Header Manipulation

    2024-02-01 14:28:01       26 阅读
  5. sqlalchemy——@listens_for

    2024-02-01 14:28:01       31 阅读
  6. 《Docker极简教程》--前言--Docker的简介

    2024-02-01 14:28:01       29 阅读
  7. SpringMVC校验注解不生效

    2024-02-01 14:28:01       32 阅读
  8. C++的学习记录

    2024-02-01 14:28:01       36 阅读
  9. 【Midjourney】新手指南:参数设置

    2024-02-01 14:28:01       29 阅读