从源码角度配合网络编程函数accept() connect()等实现的客户端服务器通信 分析下 三握手四挥手都做了什么

news2025/1/11 7:11:38

首先我们先说下网络编程API:

数据在网络上通信,通信的双方一个是 客户端, 一个是 服务器

更具体来说,不是 客户端和服务器这两个机器在  经由互联网   进行通信,

                       而是 客户端上的某一进程 与 服务器端的某一进程  进行通信。

因此,客户端与服务器间的通信  是  一个进程间通信,只不过通信的两个进程不在同一机器上。

所以进程间通信方式 实际上有 ①管道(命名管道、无名管道);②消息队列;③共享内存;④信号量;⑤信号;⑥套接字Socket 这6种方式

其中的 套接字Socket 通信就是指 不在同一机器上,需要经由网络进行通信的 进程间通信方式

【具体详见: XXXXX插入一条链接】--》链接内容就是socket 作为内河中的缓冲区,是如何被fd指向来接受梁金成穿来穿去的数据的


现在,客户端进程 与 服务器进程要通信,如何使用Socket套接字呢,需要以下流程:

不加任何IO复用技术的客户端与服务器之间数据的交换调用函数如下:

 使用epoll来监听的socket通信流程调用API如下:(图中只画了三次握手和传输数据,没画关闭连接阶段,,,)

 但,无论是哪种方式(无论是程序员自己写逻辑轮询监听文件描述符还是使用select\poll\epoll),客户端与服务器间建立连接【发生三次握手】的位置都是:connect()函数 与 accept()函数交互的位置,如下图所示:


首先,让我们了解下 三次握手四次挥手过程的状态转换 :

 连接建立阶段:

第一次握手:客户端的应用进程  向 服务器端 发出连接请求报文:发送连接请求后,客户端状态变为 "SYN_SENT"

请求报文内容为:其首部中:SYN值置为1 ;seq=x


第二次握手:服务器应用进程 响应 客户端的连接请求,向客户端发回 确认报文和连接请求 : 而后,服务器端的状态变为 "SYN_RECV"

请求报文内容为:其首部中:SYN值置为1,ACK值置为1 ;ack=x+1,seq=y。


第三次握手:客户端收到确认报文之后,通知上层应用进程连接已建立,并向服务器发出确认报文:而后,客户端状态变为 "ESTABLISHED"

请求报文内容为:其首部中:ACK值置为1,ack=y+1

(服务器端 收到 第三次握手客户端发来的确认报文,状态也变为 "ESTABLISHED")


至此,TCP连接就建立了,客户端和服务器可以愉快地玩耍了。只要通信双方没有一方发出连接释放的请求,连接就将一直保持。
 

 连接释放阶段:(假设客户端主动关闭连接)

第一次挥手:客户端发送一个断开连接包(FIN包):而后,客户端状态变为 "FIN_WAIT_1"

请求报文内容为:其首部中:FIN值置为1,ACK值置为1 ;ack=... ,seq=... 。

(当然,在fin包之前发送出去的数据,如果没有收到对应的ack确认报文,主动关闭方依然会重发这些数据)

第二次挥手:服务器端收到FIN包后,发送一个确认包给对方:而后,服务器端的状态变为 "CLOSE_WAIT"

请求报文内容为:其首部中:ACK值置为1 ;ack=... 。


第三次挥手:服务器端发送一个FIN包,告诉客户端我的数据也发送完了,不会再给你发数据了:而后,服务器端的状态变为 "LAST_ACK"

请求报文内容为:其首部中:FIN 值置为1,ACK值置为1 ;ack=...


第四次挥手:客户端 接收到 服务器端发来的FIN包,发送一个ACK给服务器端,状态转换为:"TIME_WAIT"

服务器端接收到 客户端发来的ACK包,断开与客户端的连接

客户端处于 “TIME_WAIT”状态 2MSL 时间后,也断开连接,进入"CLOSE"状态

至此,完成四次挥手




三次握手发生在 connect()函数 和  accept() 函数中,这俩函数具体干了些什么?

服务器端接收到 connect()函数实现的对服务器端的连接请求后,accept()函数与之交互:

connect()函数调用时:

由于服务器端的ip和port都已经作为地址参数传入给connect(),

因此,第一次握手时,connect()函数去封装好一个SYN包,并且在该SYN包中也写明了 seq内容、window 大小等一系列后续数据传输的参数信息,并将自己的状态置为 SYN_SEND;

第二次握手时,服务器端接收到请求:会新建一个socket (为方便称呼我们称之为 new_socket),并将客户端的ip和端口号写入该新建的socket中,而后,返回 连接请求SYN包和ACK;

第三次握手:客户端接收到应答:判断自己socket()此时状态是“SYN_SEND”,而后接受服务器端返回的ACK包,解析出ACK应答包中的通信socket的具体内容,例如window大小等,并把服务器端的ip和port写入到自己的socket中,为以后的信息传递做准备


## 一个疑问:服务器端调用accept()生成新的new_socket与客户端通信,那么,客户端访问服务器端时,它的端口号还会是客户端用于监听的socket的80端口吗?

答案是:是的,客户端在后续的数据传输中还是在访问 80端口。因为 accept 函数新创建的socket对象其实并没有进行端口的占有,而是复制了socetfd的本地IP和端口号,并且也向其中记录了连接过来的客户端的IP和端口号

## 那,多个客户端建立了多个连接请求,都在访问80端口,服务器端怎么知道那个请求时对应哪个客户端呢?

答案是:这是因为,socket不仅是一个进程间通信缓冲区,它还包含了一个用于记录控制信息的结构体,其中记录了这块缓冲区用于承担源和目的2个进程 其进程ip和端口号

因此,客户端A访问ip和80端口,服务器端对80端口的监听程序(如epoll)就会发现有数据到来,接着就会判断,这个分析这个数据包内容:

若是一个未完成三次握手连接的新客户端在发送SYN包请求连接,则,调用accept() 来新建new_socket()与之进行三次握手操作

若是一个已完成三次握手连接的客户端发送来的数据,那么就根据该数据包 则将数据放入接收缓冲区(TCP/IP协议栈 维护一个接收发送数据的缓冲区) ,当该数据包 的接受进程KK需要读数据的时候,通过调用了recv() 或read() ,进程KK根据其socket中记录的源目的ip和端口,从缓冲区中轻易找到该数据包,(并读到自己的socket空间中【这块不确定,要不别说了】)

这就是为啥服务器端 即使 就用一个端口 也 不会弄混 不同客户端发来的请求和处理的消息。

如下图所示:

服务器监听8000端口,在未建立连接时,可以看到在监听8000

在通过一个客户端建立连接后,可以看到建立了一条连接,服务器端的端口号是8000,监听的还是8000。

在连接一个客户端,可以看到建立了两条连接,服务器端都是使用8000,监听的还是8000。

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

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

相关文章

vue3+ts未使用变量报错的解决

实例 问题原因 tsconfig.json文件中开启了ts语法检查 "strict": true, // 开启严格模式,检查类型声明和赋值...是否合法 "noUnusedLocals": true, // 检查是否存在未使用的变量 "noUnusedParameters": true, // 检查是否存在会使…

app性能测试怎么做?内容全在这里了

1 app性能测试 提到APP的性能测试这个概念比较笼统,因为APP的性能测试分为服务端的性能和手机端的性能测试 1.1 app服务端性能测试 app服务端的性能测试,利用jmeter等工具模拟并发,压测服务器系统,服务端性能测试,一…

Anaconda安装-超详细版(2023)

Anaconda安装 - 超详细版(2023) 前言:彻底卸载pythonAnaconda下载地址安装详细步骤配置环境变量检验安装是否成功更改conda源(后续安装第三方库可以加快速度)超详细彻底卸载Anaconda教程Tensorflow-gpu 安装 前言&…

2023.07.29 驱动开发DAY6

通过epoll实现一个并发服务器 服务器 #include <stdio.h> #include <string.h> #include <unistd.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <sys/epoll.h…

Android getDrawable()和getColor()

Android getDrawable() 1.过时代码 虽然过时&#xff0c;但是不妨碍使用 context.getResources().getDrawable(R.drawable.xxx) 2.建议代码 context.getDrawable(R.drawable.xxx) 有API限制 3.最新代码 ContextCompat.getDrawable(getContext(), R.drawable.xxx); 有A…

C语言枚举与联合体详解

本篇文章带来枚举与联合体相关知识详细讲解&#xff01; 如果您觉得文章不错&#xff0c;期待你的一键三连哦&#xff0c;你的鼓励是我创作的动力之源&#xff0c;让我们一起加油&#xff0c;一起奔跑&#xff0c;让我们顶峰相见&#xff01;&#xff01;&#xff01; 目录 一…

Qt中文显示乱码问题

解决方法&#xff1a; 添加#pragma execution_character_set("utf-8");

Redis缓存预热

说明&#xff1a;项目中使用到Redis&#xff0c;正常情况&#xff0c;我们会在用户首次查询数据的同时把该数据按照一定命名规则&#xff0c;存储到Redis中&#xff0c;称为冷启动&#xff08;如下图&#xff09;&#xff0c;这种方式在一些情况下可能会给数据库带来较大的压力…

JavaSE - 异常

目录 异常 一. 常见的异常 1. 算数异常&#xff08;ArithmeticException&#xff09; 2. 数组越界异常&#xff08;ArrayIndexOutOfBoundException&#xff09; 3. 空指针异常&#xff08;NullPointerException&#xff09; 4. 输入不匹配异常&#xff08;InputMismatchEx…

黑马头条---day1

手机端查看 docker 容器&#xff0c;镜像操作命令 1、docker删除所有镜像命令 删除所有镜像的命令是Docker中一个非常常见的操作。下面是具体的实现步骤和命令示例&#xff1a; $ docker stop $(docker ps -aq) 停止所有正在运行的容器。 $ docker rm $(docker ps -aq) 删…

数据库应用:rsync远程同步

目录 一、理论 1.rsync 2.rsync优缺点 3.rsync三种工作模式 4.rsync同步源服务器 3. 配置rsync下行同步&#xff08;定时同步&#xff09; 4.rsync实时同步&#xff08;上行同步&#xff09; 5.配置rsync实时同步&#xff08;上行同步&#xff09; 6.使用rsync快速删除…

数组中出现次数超过一半的数字——剑指 Offer 39

文章目录 题目描述法一 哈希表法二 摩尔投票 题目描述 法一 哈希表 使用哈希映射&#xff08;HashMap&#xff09;来存储每个元素以及出现的次数。对于哈希映射中的每个键值对&#xff0c;键表示一个元素&#xff0c;值表示该元素出现的次数。 class Solution { public:int maj…

XCTF_very_easy_sql

简单的进行sql注入测试后发现不简单尝试一下按照提示 结合这句提示应该是内部访问&#xff0c;所以采用的手段应该是ssrf顺便看看包 唯一值得关注的是set-cookie说回ssrf唯一能使用的方式应该是Gopher协议找到了一个POST的python脚本 import urllib.parsepayload ""…

Linux上定位线上CPU飙高

【模拟场景】 写一个java main函数&#xff0c;死循环打印 System.out.println(“111111”) &#xff0c; 将其打成jar包放在linux中执行 1、通过TOP命令找到CPU耗用最厉害的那个进程的PID 2、top -H -p 进程PID 找到进程下的所有线程 可以看到 pid 为 94384的线程耗用cpu …

未来将会有更多基于 Cortana 的设备

在前些日子的 Build 大会首日 Keynote 中&#xff0c;微软正式确认 HP 跟 Intel 也正在开发基于 Cortana 平台的联网家居产品&#xff0c;这是继推出 Invoke 喇叭的 Harman Kardon 后&#xff0c;又有知名大牌加入到 Cortana 的阵营当中&#xff0c;有这样的品牌资源背景&#…

【Linux】-进程概念及进程状态(僵尸进程和孤儿进程)

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树&#x1f388; &#x1f389;作者宣言&#xff1a;认真写好每一篇博客&#x1f4a4; &#x1f38a;作者gitee:gitee✨ &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 动态规划算法&#x1f384; 如 果 你 …

基于中文金融知识的 LLaMA 系微调模型的智能问答系统:LLaMA大模型训练微调推理等详细教学

项目设计集合&#xff08;人工智能方向&#xff09;&#xff1a;助力新人快速实战掌握技能、自主完成项目设计升级&#xff0c;提升自身的硬实力&#xff08;不仅限NLP、知识图谱、计算机视觉等领域&#xff09;&#xff1a;汇总有意义的项目设计集合&#xff0c;助力新人快速实…

x86架构ubuntu22下运行WILL模拟器dophin

0. 环境 i5实体机ubuntu22 1. 安装依赖 $ sudo apt install build-essential git cmake ffmpeg libavcodec-dev libavformat-dev libavutil-dev libswscale-dev libevdev-dev libusb-1.0-0-dev libxrandr-dev libxi-dev libpangocairo-1.0-0 qt6-base-private-dev libblueto…

MybatisPlusInterceptor实现sql拦截器(超详细)

1 . 导入pom <dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>3.4.2</version></dependency> 2 . 配置下MybatisPlus的yml mybatis-plus:mapper-locations:- …

ssm学生贷款管理系统java助学贷银行jsp源代码mysql

本项目为前几天收费帮学妹做的一个项目&#xff0c;Java EE JSP项目&#xff0c;在工作环境中基本使用不到&#xff0c;但是很多学校把这个当作编程入门的项目来做&#xff0c;故分享出本项目供初学者参考。 一、项目描述 ssm学生贷款管理系统 系统有1权限&#xff1a;管理员…