TCP网络协议栈和Posix网络部分API总结

文章目录

  • Posix网络部分API综述
  • TCP协议栈通信过程
    • TCP三次握手和四次挥手(看下图)
      • 三次握手
        • 常见问题?
          • 为什么是三次握手而不是两次?
          • 三次握手和哪些函数有关?
          • TCP的生命周期是从什么时候开始的?
      • 四次挥手
      • 通信状态转换图
          • 由于数据包传输速率不同,client先收到了FIN,后收到ack,状态是如何转换的?
  • 核心步骤
    • 建立连接
      • socket函数
      • bind函数
      • listen函数
        • listen函数的backlog参数的作用
      • accept函数
      • connect函数
    • 数据传输
    • recv函数和send函数
    • 断开连接
      • close函数
        • 问题思考
          • 双方如果同时调用close函数会发生什么?
          • 同理双方同时调用了connect函数呢?(P2P模式会出现)

Posix网络部分API综述

  • 客户端:socket、bind(可以不绑定)、connect、send、recv、close
  • 服务端:socket、bind、listen、accept、recv、send、recv、close
  • 所有的语言进行网络通信,只要是基于linux底层都会去调用这道api。

TCP协议栈通信过程

TCP三次握手和四次挥手(看下图)

在这里插入图片描述

  • 参数解释:
    • seq:TCP头里的sequence,代表发送的TCP包的序列号。
    • ack:TCP头里的acknum,代表对上次接收到的TCP包的应答,通常是上次接收包的seq号+1。
    • SYN和ACK代表标志位。
    • syn和ack主要是实现了三个目的:
      1. 防止数据丢失,当我们发送出去的syn很久没有得到ack应答时,就会进行重传等操作。
      2. 防止数据重复,可能对方因为某些原因对一个请求发了很多次应答,可以依据这两个参数去防止数据重复接收。
      3. 防止数据乱序,这个很好理解就是根据seq和ack确保数据接收正确,因为网络包在网络上的传输速率不同,可能先发的包后面到。
    • 另外三次握手建立连接时,双方的第一个seq值是随机的。

三次握手

  1. 首先由通信双方的某一方(后面就用client了,其实server和client都是可以的)发起连接请求,即发送一个带SYN标志位的TCP数据包给server端。
  2. server端收到数据包后,除了SYN位以外,还会增加一个ACK确认位,然后发送给clinet端。
  3. client端收到数据包后,给server端回复一个带ACK确认位的数据包给服务端,到此,通信双方就算建立连接了。
常见问题?
为什么是三次握手而不是两次?
- 因为TCP连接是安全可靠的,三次握手的话会保证双方都能接受到对方的数据,第一次服务端收到客户端消息后可以确定客户端发送数据没问题,然后第二次服务端发送给客户端消息,客户端可以确定服务端的收发都没有问题,但是服务端不知道客户端接收是否成功,所以,客户端还需要发一个数据包告诉服务端,它的接收也没有问题,因此,这里是三次连接而不是两次。
三次握手和哪些函数有关?
  • 首先肯定是connect函数,调用后会发生第一次握手。
  • listen函数调用后TCP协议栈会去监听client请求,收到第一次握手后会向对方发送数据包,这是第二次握手,此时这个连接就被放入半连接队列里了。
  • 至于第三次握手是TCP协议栈完成的(其实第二次也是),收到第三次握手的数据包后,会从半连接队列里校验,校验成功后将其从半连接队列里mv到全连接队列。
TCP的生命周期是从什么时候开始的?
  • 从connect调用后就开始了。

四次挥手

  1. 第一次挥手,是主动断开连接的一方,发送带有FIN和ACK标志位的TCP包。
  2. 第二次是被动断开连接一方在收到数据包后,立马回应一个ACK标志位的TCP包,告诉对方我收到了对方的包,但是数据还没有处理完成,还有些工作要处理。
  3. 第三次依然是被动断开连接一方在数据处理完成之后,会给对方发送FIN和ACK标志位的TCP包,告诉对方我数据处理完了,你可以关闭连接了。
  4. 第四次挥手,是主动断开方发送ACK标志位的TCP包,然后被动断开方就关闭连接了。

通信状态转换图

在这里插入图片描述

  • 建立连接:
    • 服务端:
      1. 初始状态为closed状态,调用listen函数后进入LISTEN状态。
      2. 收到SYN标志位的TCP包,并向对方发送SYN和ACK标志位的TCP包后,变为SYN_RCVD状态。
      3. 收到对方带ACK标志位的包后变为ESTABLISTED状态,至此连接建立。(这里不是accept去完成三次握手的,三次握手由TCP协议栈完成,accept只是堵塞获取客户端信息的)。
    • 客户端:
      1. 初始状态为closed状态,调用connect函数,发送SYN标志位的TCP包后进入SYN_SENT状态。
      2. 由TCP协议栈完成后续的握手环节,当connect函数成功返回后,直接进入ESTABLISTED状态。
  • 断开连接:断开连接一般情况下可以是任意一方发起,这里以客户端主动断开为例
    • 服务端:
      1. 当客户端主动断开连接后,服务端收到FIN位的TCP包,此时服务端的协议栈会立刻发送给对方一个ACK确认位,此时服务端酒进入了CLOSED_WAIT状态。
      2. 此时recv函数会继续去处理数据,当读到数据长度为0,并且数据全部处理完后,服务端主动调用close函数关闭通信fd,这时服务端会给客户端发送FIN标志位的TCP包,然后服务端进入LAST_ACK状态。
      3. 当服务端收到客户端的ACK标志位的包后,服务端由LAST_ACK转变为CLOSED状态。
    • 客户端:
      1. 客户端主动调用closed函数关闭fd,那么TCP协议栈会给服务端发送一个带SYN标志位的TCP包,此后客户端进入FIN_WAIT1状态。
      2. 当收到服务端带ACK标志位的TCP包后,由FIN_WAIT1转变为FIN_WAIT2状态。
      3. 当收到服务端带FIN标志位的TCP包,并向服务端发送ACK标志位的TCP包后就进入了TIME_WAIT状态,该状态会持续2MSL,(MSL为网络包在网络中的最大存活时间。),当2MSL到达后,转换为CLOSED状态。
  • 注意:
    • TCP的状态转换是由TCP协议栈管理的,不是由代码管理。
由于数据包传输速率不同,client先收到了FIN,后收到ack,状态是如何转换的?
  • 就是上图的由FIN_WAIT1状态直接到了CLOSING状态,然后等收到了ack后再转换为TIME_WAIT。

核心步骤

建立连接

socket函数

  • socket函数做了两件事情:
    1. 分配一个fd用来监听连接,这里是采用bit_map的算法分配fd,当对应fd位为0时就可以分配。
    2. 创建一个TCP控制块TCB。

bind函数

  • 其实bind函数就做了一个操作,将ip和port设置到TCP控制块里的src ip和port。

listen函数

  • listen函数就像一个开关,如果不调用listen函数的话,是没有办法进行三次握手的。
  • listen函数也做了两个事情:
    1. 将tcb里的status设置为LISTEN状态。
    2. 为tcb分配分配一个全连接队列和半连接队列。(全连接队列指的是已经完成三次握手的,半连接队列指还没有完成三次握手的)
listen函数的backlog参数的作用
  • 起初主要是为了防止syn(第一次握手)泄洪,但是现在一般有堡垒机和防火墙,这个参数不常用了。
  • listen函数的backlog参数从70年代发展到现在大概有三个版本:
    1. 指的是半连接队列的长度。
    2. 半连接+全连接队列的长度。
    3. 全连接队列的长度。(如今的版本)

accept函数

  • accept函数做了两件事情:
    1. 为该连接分配一个fd
    2. 将fd绑定到从全连接队列取出来的连接tcb上,这个用于通信的tcb是个五元组(src ip和port,dst ip和port,传输协议)和读写缓冲区。
  • 注意:若listen的sockfd被设置为ET模式,那么需要循环调用accept去获取fd。

connect函数

  • 向对方发送带SYN标志的数据包,请求连接。

数据传输

recv函数和send函数

  • 这两个函数是将数据从TCP缓冲区读到用户buffer里以及将用户数据写到TCP缓冲区里,而不是直接端到端的同步通信,这里是异步的,数据什么时候从缓冲区发出去,怎么发和send没有关系

断开连接

close函数

  • close函数通信双方都可以调用,也做了两个事情:
    1. 关闭fd。
    2. 告诉TCP协议栈,向对方发送一个带FIN标志位的数据包。
  • server调用recv返回值为0的时候,就意味着收到了对方的FIN数据包,且TCP协议栈自动向对方发送ACK确认。(第一次挥手)此时server也可以调用close函数去关闭fd,然后server会向对方也发送带FIN的数据包。
问题思考
双方如果同时调用close函数会发生什么?
  • 首先双方都向对方发送FIN数据包,然后双方收到后,都向对方发送ACK确认,然后双方又向对方发送FIN,最后双方都收到了对方的ack,此时双方都产生了time_wait状态。(服务器一般情况下不要去调用close函数,减少time_wait状态出现)
同理双方同时调用了connect函数呢?(P2P模式会出现)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/503689.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

git基本操作二(小白快速上手)

1、前言 接上篇我们接着来继续讲 2、.gitignore忽略文件 创建一个.gitignore文件,并将其置于项目的根目录下,Git将自动识别并根据该规则忽略相应的文件和目录。 # 忽略所有的 .log 文件 *.log# 但跟踪所有的 build.log 文件 !build.log# 忽略所有的 /lo…

lookup函数

lookup函数 单条件查询 示例 扩展多条件 扩展

文件的顺序读写——顺序读写函数——fgets、fgetc、fputs、 fputc

✨✨ 欢迎大家来到莉莉的博文✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 目录 一、fgetc和fputc函数 1.1 fputc 1.2 fgetc 二、fputs和fgets函数 2.1 fputs函数 2.2 fgets函数 一、fgetc和fputc函数 1.1 fputc 返回类…

结构体类型,结构体变量的创建和初始化 以及结构中存在的内存对齐

一般结构体类型的声明 struct 结构体类型名 { member-list; //成员表列 }variable-list; //变量表列 例如描述⼀个学⽣: struct Stu { char name[20]; //名字 int age; //年龄 char sex[5]; //性别 }; //结构体变量的初始化 int main() { S…

鸿蒙OS开发实例:【工具类封装-页面路由】

import common from ohos.app.ability.common; import router from ohos.router 封装app内的页面之间跳转、app与app之间的跳转工具类 【使用要求】 DevEco Studio 3.1.1 Release api 9 【使用示例】 import MyRouterUtil from ../common/utils/MyRouterUtil MyRouterUtil…

基于重写ribbon负载实现灰度发布

项目结构如下 代码如下&#xff1a; pom&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocat…

使用第三方远程连接工具ssh连接vagrant创建的虚拟机

vagrant默认密码都是vagrant 密码认证默认是关闭的&#xff0c;进入虚拟机&#xff0c;打开密码认证 1、使用命令vi /etc/ssh/sshd_config进入配置&#xff0c;注意要切换到root用户&#xff0c;这个配置root有权限 2、找到PasswordAuthentication默认为no,改为yes 3、重启虚…

ETL中RESTful API 组件的用法

一、ETL是什么 ETL&#xff0c;全称为Extract-Transform-Load&#xff0c;即数据提取&#xff08;Extract&#xff09;、数据转换&#xff08;Transform&#xff09;和数据加载&#xff08;Load&#xff09;。这是数据仓库中数据处理的重要过程。ETL过程中&#xff0c;数据从源…

小小狠招:巧妙使用HANA数据库的jdbc driver

SAP旗下的HANA数据库&#xff0c;实际上是分为两个系列进行发布&#xff0c;一种是基于本地部署的称之为HANA Platform。另一种是面向Cloud平台的&#xff0c;称之为HANA Cloud。 在实际使用当用&#xff0c;因为两者基本上共用同一代码库&#xff0c;除个别地方略有差异以外&…

【更清晰】照片分享,欢迎家庭新成员HPE ProLiant DL580 Gen9

正文共&#xff1a;1234 字 29 图&#xff0c;预估阅读时间&#xff1a;1 分钟 距离上一台服务器HPE ProLiant DL360 Gen9开箱已经过去4年了&#xff0c;回忆满满&#xff08;风雨同舟&#xff0c;感谢HP Proliant DL360 Gen9陪我走过的四年&#xff09;&#xff1b;就在上周&a…

相册清理大师-手机重复照片整理、垃圾清理软件

相册清理大师是一款超级简单实用的照片视频整理工具。通过便捷的操作手势&#xff0c;帮助你极速整理相册中的照片和视频、释放手机存储空间。 【功能简介】 向上滑动&#xff1a;删除不要的照片 向左滑动&#xff1a;切换下一张照片 向右滑动&#xff1a;返回上一张照片 整理分…

拌合楼管理软件开发(十三) 对接耀华XK3190-A9地磅(实战篇)

前言: 实战开整 目前而言对于整个拌合楼管理软件开发,因为公司对这个项目还处于讨论中,包括个人对其中的商业逻辑也存在一些质疑,都是在做一些技术上的储备.很早就写好了串口与地磅对接获取代码,也大概知道真个逻辑,这次刚好跟库区沟通,远程连接到磅房电脑,开始实操一下. 一、地…

Sql注入---基础

文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 一.Sql注入概述 攻击者通过构造恶意的SQL查询语句&#xff0c;将其注入到应用程序的数据库查询中&#xff0c;以执行未经授权的操作或者获取敏感信息。 假设如下场景&#xff0c;当你想要知道对…

双端队列的插入与删除操作的实现及其时间复杂度分析

双端队列(deque,全称为double-ended queue)是一种支持在两端插入和删除元素的数据结构。与栈和队列不同,双端队列提供了更加灵活的操作方式。在实现双端队列时,我们可以采用数组作为底层数据结构,以保证插入和删除操作的时间复杂度为O(1)。 一、双端队列的基本概念 双…

《QT实用小工具·四》屏幕拾色器

1、概述 源码放在文章末尾 该项目实现了屏幕拾色器的功能&#xff0c;可以根据鼠标指定的位置识别当前位置的颜色 项目功能包含&#xff1a; 鼠标按下实时采集鼠标处的颜色。 实时显示颜色值。 支持16进制格式和rgb格式。 实时显示预览颜色。 根据背景色自动计算合适的前景色…

Artplayer视频JSON解析播放器源码|支持弹幕|json数据模式

全开源Artplayer播放器视频解析源码&#xff0c;支持两种返回模式&#xff1a;网页播放模式、json数据模式&#xff0c;json数据模式支持限制ip每分钟访问次数UA限制key密钥&#xff0c;也可理解为防盗链 &#xff0c;本播放器带弹幕库。 运行环境 推荐使用PHP8.0 redis扩展…

【每日跟读】常用英语500句(400~500)

【每日跟读】常用英语500句 Where can I buy a ticket? 在哪里能买到票&#xff1f; When is the next train? 下趟火车什么时候到&#xff1f; Thank you so much for helping me move yesterday. 非常感谢你昨天帮我搬家 I’m feeling a little under the weather toda…

专升本-信息安全

信息安全&#xff1a; 1.信息安全的基本属性&#xff1a;保密性&#xff0c;完整性&#xff0c;可用性 信息本身的安全是指保证信息的保密性&#xff08;非授权用户不能访问信息&#xff09;&#xff0c;完整性&#xff08;信息正确&#xff0c;完整&#xff0c;违背篡改&…

从0开始搭建基于VUE的前端项目

准备与版本 安装nodejs(v20.11.1)安装vue脚手架(vue/cli 5.0.8) ,参考&#xff08;https://cli.vuejs.org/zh/&#xff09;vue版本&#xff08;2.7.16&#xff09;&#xff0c;vue2的最后一个版本 初始化项目 创建一个git项目&#xff08;可以去gitee/github上创建&#xff…

CVE-2023-4427:Out-of-bounds access in ReduceJSLoadPropertyWithEnumeratedKey

文章目录 前言环境搭建for-in && enum cache漏洞分析漏洞利用总结参考 前言 之前分析调试漏洞时&#xff0c;几乎都是对着别人的 poc/exp 调试&#xff0c;感觉对自己的提升不是很大&#xff0c;所以后面分析漏洞时尽可能全面分析&#xff0c;从漏洞产生原理、如何稳定…