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模式会出现)

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HarmonyOS实战开发-如何实现一个自定义抽奖圆形转盘

介绍 本篇Codelab是基于画布组件、显式动画,实现的一个自定义抽奖圆形转盘。包含如下功能: 通过画布组件Canvas,画出抽奖圆形转盘。通过显式动画启动抽奖功能。通过自定义弹窗弹出抽中的奖品。 相关概念 Stack组件:堆叠容器&am…

从0开始搭建基于VUE的前端项目(一) 项目创建和配置

准备与版本 安装nodejs(v20.11.1)安装vue脚手架(@vue/cli 5.0.8) ,参考(https://cli.vuejs.org/zh/)vue版本(2.7.16),vue2的最后一个版本vue.config.js的配置详解(https://cli.vuejs.org/zh/config/)element-ui(2.15.14)(https://element.eleme.io/)vuex(3.6.2) (https://…

K8S命令行可视化实验

以下为K8s命令行可视化工具的实验内容,相比于直接使用命令行,可视化工具可能更直观、更易于操作。 Lens Lens是用于监控和调试的K8S IDE。可以在Windows、Linux以及Mac桌面上完美运行。在 Kubernetes 上: 托管地址:github/lensa…

基于easyx库的C/C++游戏编程实例-飞机大战

飞机大战游戏设计 首先创建飞机/子弹结构: struct Plane {int x;int y;bool live;int width;int height;int type;int hp; }player,bul[BUL_NUM],enemy[ENE_NUM];你需要加载图片: void ImageLoad() {//背景loadimage(&bg[0], "./image/飞机大…

机器人运动控制

一、基础 1.1 矢量速度和旋转速度 矢量速度用来控制运动方向,任何一个方向都可以看成x、y、z三轴方向的合。单位规定是m/s。 旋转速度用来控制旋转方向,可以看成x、y、z三轴方向旋转的合。单位规定是pi/s。 速度消息包,可以在ROS Index上搜…

助力福建新型职业农民培育 北方天途推进无人机植保应用培训

为加强新型职业农民的职业培育,扩展新型农民的知识范围和专业技术水平,推进农业供给侧结构性改革。日前,在农业部门的大力支持下,北方天途航空和宁德天禾科技服务携手为福建省农民朋友开展了植保无人机驾驶员的应用培训。福建省农…

同步复位和异步复位的优缺点

同步复位 优点:能确保电路是100%的; 同步复位可以综合处更小的触发器; 可以保证复位只发生在有效时钟边沿,过滤掉复位信号毛刺; 内部逻辑产生的复位信号,采用同步复位可以有效过滤掉毛刺。 缺点&#xff1a…

网页布局案例 浮动

这里主要讲浮动 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title><style>*{padding: 0;margin: 0;}.header{height: 40px;background-color: #333;}.nav{width: 1226px;heig…

深入理解数据结构(2):顺序表和链表详解

文章主题&#xff1a;顺序表和链表详解&#x1f331;所属专栏&#xff1a;深入理解数据结构&#x1f4d8;作者简介&#xff1a;更新有关深入理解数据结构知识的博主一枚&#xff0c;记录分享自己对数据结构的深入解读。&#x1f604;个人主页&#xff1a;[₽]的个人主页&#x…

单片机串口通过查询接收数据,同时具备接收超时、异常判断,常规通用处理算法实现

单片机串口通过查询接收数据,同时具备接收超时、异常判断,常规通用处理算法实现;1、实现过程 (1)初始化串口:首先,需要配置串口的基本参数,如波特率、数据位、停止位和校验位等。这通常涉及到GPIO引脚的配置(如PA2为发送引脚,PA3为接收引脚),以及USART的初始化。 …

Day38 动态规划

Day38 动态规划 509. 斐波那契数 我的思路&#xff1a; 很简单的递归 解答&#xff1a; class Solution {public int fib(int n) {if(n 0 || n 1) {return n;}return fib(n - 1) fib(n - 2);} }70. 爬楼梯 我的思路&#xff1a; 看了题解才明白本质上还是第一题的斐波那…

机器学习——降维算法-奇异值分解(SVD)

机器学习——降维算法-奇异值分解&#xff08;SVD&#xff09; 在机器学习中&#xff0c;降维是一种常见的数据预处理技术&#xff0c;用于减少数据集中特征的数量&#xff0c;同时保留数据集的主要信息。奇异值分解&#xff08;Singular Value Decomposition&#xff0c;简称…

csp资料

头文件 #include <bits/stdc.h> using namespace std isdigit(c); isalpha(c); switch(type){case value : 操作 } continue;//结束本轮循环 break;//结束所在的整个循环tips: //除法变乘法来算 //减法变加法 num1e42;//"1e4"表示10的4次方//用于移除容器中相…

某国投集团知识竞赛活动方案

一、抽签分组办法 1.抽签&#xff1a;参赛队伍赛前进行抽签分组。 2.分组&#xff1a;全部报名参赛队伍按照抽签顺序分为4组&#xff0c;每组7支队伍进行预赛&#xff0c;9月16日上午1、2组进行初赛&#xff0c;9月16日下午3、4组进行初赛。每组决出的前三名进入决赛。 二、初…

蓝桥杯2014年第十三届省赛真题-切面条

一、题目 切面条 一根高筋拉面&#xff0c;中间切一刀&#xff0c;可以得到2根面条。 如果先对折1次&#xff0c;中间切一刀&#xff0c;可以得到3根面条。 如果连续对折2次&#xff0c;中间切一刀&#xff0c;可以得到5根面条。 那么&#xff0c;连续对折10次&#xff0c;中间…

并查集进阶——带权并查集

带权并查集是在并查集的基础上加入数组来记录某些值&#xff0c;比如说当前家族的人数 原理也较简单&#xff0c;直接在合并时相加赋值就可以了&#xff0c;但是不要忘记初始化 例题一.合并家族 给定一个数n&#xff0c;代表总人数&#xff0c;一开始每个人自己是一家&#x…

二维码门楼牌管理应用平台建设:引领现代化小区管理新篇章

文章目录 前言一、二维码门楼牌管理应用平台概述二、三维动态单体化技术的优势三、二维码门楼牌管理应用平台的应用场景四、展望未来 前言 随着城市化的快速推进&#xff0c;现代化小区如雨后春笋般涌现&#xff0c;对小区管理的效率和智能化提出了更高要求。二维码门楼牌管理…

代码随想录Day39

Day 39 动态规划 part02 今日任务 62.不同路径 不同路径 II 代码实现 62.不同路径 public int uniquePaths(int m, int n) {if (m < 1 || n< 1) return 1;int[][] dp new int[m][n];for (int i 0; i < dp.length; i) {for (int j 0; j < dp[i].length; j) …

鸿蒙OS开发实例:【瀑布流式图片浏览】

介绍 瀑布流式展示图片文字&#xff0c;在当前产品设计中已非常常见&#xff0c;本篇将介绍关于WaterFlow的图片浏览场景&#xff0c;顺便集成Video控件&#xff0c;以提高实践的趣味性 准备 请参照[官方指导]&#xff0c;创建一个Demo工程&#xff0c;选择Stage模型熟读Har…

【算法】基数排序

简介 基数排序&#xff08;*Radix sort&#xff09;是一种非比较排序算法&#xff08;non-comparative sorting algorithm&#xff09;。现代计算机的基数排序算法由 计数排序 算法的开发人哈罗德H西华德&#xff08;Harold H. Seward&#xff09;于1954年于麻省理工大学开发。…