当前位置: 首页 > article >正文

趣讲TCP三次握手

一、TCP三次握手简介

TCP(Transmission Control Protocol,传输控制协议)是一种面向连接的、可靠的、基于字节流的传输层通信协议。在TCP连接中,只有两方进行通信,它使用校验和、确认和重传机制来保证数据的可靠传输。

二、三次握手过程

1.第一次握手(SYN)

客户端向服务器发送一个SYN(同步序列编号)标志位的TCP数据包,请求建立连接。这个数据包中包含了客户端的初始序列号(ISN)。

2.第二次握手(SYN+ACK)

服务器收到客户端的SYN请求后,会回复一个带有SYN和ACK(确认)标志位的数据包,称为SYN-ACK响应。这个响应中,服务器确认了客户端的SYN请求,并指定了服务器的初始序列号(ISN)。同时,服务器还会对客户端的初始序列号进行确认,即发送一个确认号(ACK号),表示已经收到客户端发送的序列号加1的数据。

3.第三次握手(ACK)

客户端收到服务器的SYN-ACK响应后,会发送一个带有ACK标志位的数据包,表示确认了服务器的响应。这个ACK数据包中,客户端会确认收到了服务器的SYN响应,并指定了下一个要发送的序列号(即服务器的初始序列号加1)。至此,三次握手完成,TCP连接建立成功,双方可以开始进行数据传输。

三、为什么需要三次握手

1.三次握手原因

三次握手的主要目的是为了确保双方的发送和接收能力都正常,防止已失效的连接请求报文突然又传送到服务器,从而产生错误。通过三次握手,可以告知对方自己的初始序号值,并确认收到对方的初始序号值。

  • 第一次握手:客户端向服务器端发送报文,证明客户端的发送能力正常。
  • 第二次握手:服务器端接收到报文并向客户端发送报文,证明服务器端的接收和发送能力正常。
  • 第三次握手:客户端向服务器发送报文,证明客户端的接收能力正常。

这样,三次握手确保了双方都准备好进行数据传输,从而可以愉快地进行通信了。

2.无三次握手示例

如果没有三次握手,TCP连接将无法建立,因为三次握手确保了双方都准备好发送和接收数据。下面我将提供一个简化的示例,展示如果没有三次握手,客户端和服务器之间的通信可能会出现的问题。

2.1服务器端代码(没有三次握手)

import socket

def main():
    # 创建 socket 对象
    server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    # 获取本地主机名
    host = socket.gethostname()
    port = 9999

    # 绑定端口号
    server_socket.bind((host, port))

    # 设置最大连接数,超过后排队
    server_socket.listen(5)
    print("服务器启动,等待连接...")

    while True:
        # 建立客户端连接
        client_socket, addr = server_socket.accept()
        print(f"连接地址: {str(addr)}")

        # 直接发送数据,没有等待客户端的确认
        client_socket.send(b'你好,客户端!')

        # 关闭连接
        client_socket.close()

if __name__ == "__main__":
    main()

2.2客户端代码(没有三次握手)

import socket
import time

def main():
    # 创建 socket 对象
    client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

    # 获取本地主机名
    host = socket.gethostname()
    port = 9999

    # 连接到服务器,指定主机和端口
    client_socket.connect((host, port))

    # 等待一段时间,模拟没有发送SYN和等待服务器的SYN+ACK
    time.sleep(2)

    # 接收数据,但由于没有完成三次握手,服务器可能还没有准备好发送数据
    data = client_socket.recv(1024)
    if data:
        print(f"来自服务器的消息:{data.decode()}")
    else:
        print("没有收到任何数据")

    # 关闭连接
    client_socket.close()

if __name__ == "__main__":
    main()

2.3可能出现的问题

  1. 数据丢失:在客户端尝试接收数据之前,服务器可能还没有准备好发送数据,或者服务器在客户端准备好接收之前就已经发送了数据。这可能导致客户端接收不到任何数据。

  2. 连接不稳定:由于没有进行三次握手,客户端和服务器之间的连接可能不稳定,容易出现断开连接的情况。

  3. 数据乱序:没有三次握手,就无法确保数据的顺序和完整性,可能导致数据乱序或丢失。

  4. 重复数据:如果客户端或服务器在没有确认对方已准备好的情况下发送数据,可能会导致重复发送数据。

这些代码示例仅用于说明没有三次握手可能导致的问题,实际应用中,TCP协议会自动处理三次握手的过程,开发者不需要手动实现。在实际编程中,我们通常使用高级的网络库,这些库已经为我们处理了这些底层的细节。

四、三次握手示例代码

1.服务器端代码(Server.py)

import socket

# 创建 socket 对象
server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 获取本地主机名
host = socket.gethostname()
port = 9999

# 绑定端口号
server_socket.bind((host, port))

# 设置最大连接数,超过后排队
server_socket.listen(5)

while True:
    # 建立客户端连接
    client_socket, addr = server_socket.accept()
    print(f"连接地址: {str(addr)}")

    # 接收小于 1024 字节的数据
    data = client_socket.recv(1024).decode()
    print(f"收到消息:{data}")

    # 发送数据
    client_socket.send(b'服务器收到消息')

    # 关闭连接
    client_socket.close()

2.客户端代码(Client.py)

import socket

# 创建 socket 对象
client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 获取本地主机名
host = socket.gethostname()
port = 9999

# 连接到服务器,指定主机和端口
client_socket.connect((host, port))

# 发送数据
client_socket.send(b'你好,服务器!')

# 接收小于 1024 字节的数据
data = client_socket.recv(1024)
print(f"来自服务器的消息:{data.decode()}")

# 关闭连接
client_socket.close()

3.运行步骤

  1. 首先运行服务器端代码(Server.py),它会在本地的9999端口上监听客户端的连接。
  2. 然后运行客户端代码(Client.py),它会尝试连接到服务器,并发送一条消息。
  3. 服务器接收到消息后,会回复一条消息给客户端。
  4. 客户端接收到服务器的回复,并打印出来。
  5. 最后,客户端和服务器都会关闭连接。

http://www.kler.cn/a/564479.html

相关文章:

  • 使用Apifox动态生成请求参数
  • 如何将mobaxterm的默认编辑器修改为vscode
  • 深入浅出:插入排序算法完全解析
  • Java中,Scanner和System.out超时的解决方法及原理
  • 【含文档+PPT+源码】基于过滤协同算法的旅游推荐管理系统设计与实现
  • Spring Boot + JSqlParser:全面解析数据隔离最佳实践
  • 左值引用与右值引用详解
  • Autosar_RTE基础概念整理
  • 为AI聊天工具添加一个知识系统 之125 详细设计之66 智能语义网络
  • leetcode 912. 排序数组
  • 需求: 使用 minio 做一个 企业微信对话的下载、存储,利用deepseek进行对话回复
  • 在VSCode中安装jupyter跑.ipynb格式文件
  • PhpStorm 绿色版 安装包 Win/Mac/Linux 商业的PHP集成开发环境 2025全栈开发终极指南:从零配置到企业级实战
  • 腾讯云的海外轻量云套餐
  • 【学写LibreCAD】3 qmetaobject库介绍
  • Pytorch加载数据的Dateset类和DataLoader类
  • .hive-staging_hive临时文件处理
  • 【人工智能】数据挖掘与应用题库(201-300)
  • Python连接SQL SEVER数据库全流程
  • python 网络安全常用库 python做网络安全