怎么介绍网站的优缺点/域名关键词排名查询
6. TCP 三次握手四次挥手
HTTP 协议是 Hype Transfer Protocol(超文本传输协议)的缩写,是用于从万维网(WWW:World Wide Web)服务器(sever)传输超文本到客户端(本地浏览器 client)的传送协议。HTTP 协议基于 TCP/IP 协议之上,HTTPS 基于 TLS/SSL 协议层上,两者都是属于应用层的面向对象的协议。
由上图可知,HTTP 协议工作前需要 client 与 sever 建立连接。该连接由 tcp 来完成,tcp 与 ip 共同组成了 Internet,也就是著名的 TCP/IP 通信协议。
6.1 TCP 简介
TCP(Transmission Control Protocol)全名传输控制协议,是主机对住几层的传输控制协议,提供可靠的连接服务,采用三次握手来建立一个连接。与 UDP 都是传输层的协议,比 UDP 更可靠,默认端口 80 。
TCP标志位(位码):
- SYN(synchronous):建立连接
- ACK(acknowledgement):确认
- ack:确认号
- PSH(push):传送
- FIN(finish):结束
- RST(reset):重置
- URG(urgent):紧急
- Sequence number:顺序号码
- Acknowledge number:确认号码
6.2 三次握手
最初两端 TCP 进程都处于关闭状态,client 主动打开连接,server 被动打开连接。大致步骤:client、server 关闭 —— server 收听到 listen —— client 同步已发送状态 SYN-SENT —— server 同步收到状态 SYN_RCVD —— client、server 已建立状态 ESTABLEISHED。
规定:SYN=1 的报文不传输数据,并消耗一个随机序列号。
1、第一次
client 向 server 发送连接请求报文 SYN=1
,同时生成初始序列化 seq=x
,此时 client 进入 SYN-SENT
(同步已发送)状态。
2、第二次
server 收到请求报文后,如果同意连接,则发出确认报文。确认报文中包含:ACK=1,SYN=1
,确认号:ack=x+1(即上一次的seq+1)
,同时也要为自己随机初始化一个序列号 seq=y
。此时 server 进入 SYN-RCVD
(同步收到)状态。这个报文也没有携带数据,循环 client 是否准备好。
3、第三次
client 收到确认后,向 server 给出确认:ACK=1(与 server 给出的一致),ack=y+1
,此时 client 连接建立,进入 ESTABLISHED
状态。这里客户端表示已经准备好了。
6.3 四次挥手
1、第一次
client 发送一个 FIN ,用来结束连接。client 进程发出连接释放报文,并停止发送数据。释放报文首部:FIN=1
,序列号 seq=i
。
此时 client 进入 FIN_WAIT_1
(终止等待1)状态。
2、第二次
server 收到这个 FIN 后,返回一个 ACK
(确认),确认序号:ack=i+1
。同时携带自己的序列号 seq=j
。
此时, server 进入 CLOSED_WAIT
(关闭等待)状态。
并通知高层的应用进程,此时处于半关闭状态,client 没有数据发送了,但 server 若发送数据,client 依然会接收,这种状态还会持续一段时间。
3、第三次
server 将最后的数据发送完毕后,发送一个 FIN
(结束),确认序号:ack=i+1
,同时携带序号 seq=w
,准备 关闭 client 的连接,等待 client 的最后确认。
此时,server 进入 LAST_ACK
(最后确认)状态。
4、第四次
client 发送 ACK
确认,并将确认序号+1:ack=w+1
,而自己序列号 seq=i+1
。
此时,client 进入 TIME_WAIT
(时间等待)状态。
**Note:**此时 client 并没有释放,必须等待 2MSL(最长报文段寿命)使君子后,当 server 撤销相应 TCB 后,从进入
CLOSED
状态。server 只要收到了 client 发出的确认,立即进入CLOSED
状态。同样,撤销TCB后,就结束了这次的TCP连接
为什么会是四次挥手?
三次握手时没有数据传输,而四次挥手时涉及到有数据传输。client 发出关闭请求,表示已经数据传输完毕,但是 server 有可能数据还未传输完毕,这时就需要已 server 端数据是否传输完毕为标准,因此需要四次。
当高并发时,现实情况往往是 server 先断开 client 连接,因为多保存 client 一次连接,就会多占用一些资源。因此在短时间内再次向 server 发起连接,会提示 serve time_wait。
客户端突然挂掉了怎么办?
正常连接时,客户端突然挂掉了,如果没有措施处理这种情况,那么就会出现客户端和服务器端出现长时期的空闲。解决办法是在服务器端设置保活计时器,每当服务器收到
客户端的消息,就将计时器复位。超时时间通常设置为2小时。若服务器超过2小时没收到客户的信息,他就发送探测报文段。若发送了10个探测报文段,每一个相隔75秒,
还没有响应就认为客户端出了故障,因而终止该连接。
参考文章:https://www.cnblogs.com/qdhxhz/p/8470997.html
7. 客户端服务端循环发送信息
之前设计的 socket 程序只能进行一次发送接收就终止掉了,而现实情况不可能只有一次发送与接收,往往都是循环往复,那么就需要给 socket client 和 socket server 添加循环机制。
服务端:
from socket import *ip_port = ('127.0.0.1', 8000)
back_log = 5
buffer_size = 1024tcp_server = socket(AF_INET, SOCK_STREAM)
tcp_server.bind(ip_port)
tcp_server.listen(back_log)print('服务端开始运行')
conn, addr = tcp_server.accept()while True:data = conn.recv(buffer_size)print('客户端发来的信息是', data.decode('utf-8'))conn.send(data.upper())conn.close()
tcp_server.close()
服务端开始运行
客户端发来的信息是 python
客户端:
from socket import *ip_port = ('127.0.0.1', 8000)
buffer_size = 1024tcp_client = socket(AF_INET, SOCK_STREAM)
tcp_client.connect(ip_port) # 连接服务端while True:msg = input('请输入要发送的信息:')tcp_client.send(msg.encode('utf-8'))print('---------客户端已经发送消息------------')data = tcp_client.recv(buffer_size)print('接收到服务端发来的信息', data.decode('utf-8'))tcp_client.close()
请输入要发送的信息:python
---------客户端已经发送消息------------
接收到服务端发来的信息 PYTHON
请输入要发送的信息:
8. socket 收发信息原理剖析
socket 客户端和服务端都属于应用层,即用户态层面。它们产生的数据(或从客户端发送到服务端的数据)必须通过内核态调用操作系统,将数据 copy 到内存中(缓存),然后根据 TCP/UDP 协议、通过网卡、Internet 传输到服务端。
服务端再通过内核态从缓存中取出数据。收发消息会在缓存中形成一个消息队列,遵循 先进先出原则,后面进来的消息后处理。
操作系统的体系架构分为 用户态和内核态,内核从本质上讲也一种软件 —— 控制计算机的硬件资源,并提供上层应用程序运行的环境。
用户态即上层应用程序的活动空间,应用程序的执行必须依托内核提供的资源,包括(CPU、存储、I/O资源等)。为了使用户态能访问这些资源,内核必须为上层应用提供访问的接口 —— 系统调用(系统调用是操作系统的最小单位)。
9. 服务端循环连接请求来接收信息
9.1 当用户输入为空或直接回车时
当用户在 client 端输入为空,或直接输入回车时,client 端与 server 端都开在接收信息处。这是因为 client 端没有真正的信息(0 字节)发送给 server 端,因此 server 端就不会有信息回复。
解决办法:在 client 端对用户输入的信息进行判断即可
if not msg:continue # 如果输入信息为空,那么继续输入
9.2 当 client 端异常断开时
当我们直接断开 client 的连接,而非四次挥手时正常断开,发现 server 直接报如下错误:
ConnectionResetError: [WinError 10054] 远程主机强迫关闭了一个现有的连接。
现实情况中,不可能一个 client 端每次不正常断开连接,就导致 server 端直接断开。那么其他的 client 就不能连接 server。
解决办法:对 server 端信息接收处使用异常处理
while True:try:data = conn.recv(buffer_size)print('客户端发来的信息是', data.decode('utf-8'))conn.send(data.upper())except Exception:break
这样不论 client 端是怎么断开的,都不会导致 server 端断开。
9.3 当有多个 client 发起连接时
当有多个 client 发起连接时,遵循 先进先出 原则。先连接的 client ,先处理,后发起连接的 client 会被存储到 back_log 中。back_log 为链接监听(listen)的最大数目。
每次只能处理一天连接,当处理完毕后就会直接关闭连接,也就是说只能服务一个 client,我们希望的是 server 端能够循环提供服务,显然这不是我们想要的结果。
解决办法:对被动接受 client 的连接处进行循环(即 accept)
服务端:
from socket import *# 获取主机名
host = gethostname()
# 端口号
port = 8080back_log = 5
buffer_size = 1024tcp_server = socket(AF_INET, SOCK_STREAM)
tcp_server.bind((host, port))
tcp_server.listen(back_log)while True: # 循环接收 client 发起的连接print('服务端开始运行')print(host)conn, addr = tcp_server.accept()while True: # 循环接收 client 发来的信息,以及发送信息给 client try: # 对 client 的异常断开进行异常处理data = conn.recv(buffer_size)print('客户端发来的信息是', data.decode('utf-8'))conn.send(data.upper())except Exception:breakconn.close()
tcp_server.close()
客户端:
from socket import *# 获取主机名
host = gethostname()
# 端口号
port = 8080buffer_size = 1024tcp_client = socket(AF_INET, SOCK_STREAM)
tcp_client.connect((host, port)) # 连接服务端while True:msg = input('请输入要发送的信息:').strip()if not msg: continue # 对用户输入的信息进行判断tcp_client.send(msg.encode('utf-8'))print('---------客户端已经发送消息------------')data = tcp_client.recv(buffer_size)print('接收到服务端发来的信息', data.decode('utf-8'))tcp_client.close()
9.5 总结
要想 client 与 server 能够自由交互数据,并且 server 能循环提供服务,需要满足如下条件:
- 需要对用户输入的数据进行判断
- server 能够处理 client 异常断开时的情况
- server 要能够循环接收 client 发起的连接
10. socket 函数
1. 服务端套接字函数
- s.bind():绑定(主机,端口号)到套接字,元组形式
- s.listen():开始 TCP 监听
- s.accept():被动接受 TCP 客户的连接,(阻塞式)等待连接的到来。
2. 客户端套接字函数
- s.connect():主动初始化 TCP 服务器连接
- s.connect_ex():connect()函数的拓展版本,出错时返回出错码,而不是抛出异常
3. 公共用途套接字函数
- s.recv():接受 TCP 数据
- s.send():发送 TCP 数据(send 在待发送数据量大于己端缓存区剩余空间时,数据丢失,不会发完)
- s.sendall():发送完整的 TCP 数据(本质就是循环调用 send,sendall 在待发送数据量大于己端缓存区剩余空间时,数据不丢失,循环调用 send 直到发完。)
- s.recvfrom():接收 UDP 数据
- s.sendto():发送 UDP数据
- s.getpeername():连接到当前套接字的远端的地址
- s.getsockname():当前套接字的地址
- s.getsockopt():返回指定套接字的参数
- s.setsockopt():设置指定套接字的参数
- s.close():关闭套接字
面向锁的套接字函数
- s.setblocking():设置套接字的阻塞与非阻塞模式
- s.settimeout():设置阻塞套接字操作的超时时间
- s.gettimeout():得到阻塞套接字操作的超时时间
面向文件的套接字的函数
- s.fileno():套接字的文件描述符
- s.makefile():创建一个与该套接字相关的文件
**Tips:**send 一次最大数据最好控制在 8 k 左右,为了避免超过内存大小,可以使用 sendall 方法。其本质是在循环调用 send 方法。
相关文章:

网络编程(二)
6. TCP 三次握手四次挥手 HTTP 协议是 Hype Transfer Protocol(超文本传输协议)的缩写,是用于从万维网(WWW:World Wide Web)服务器(sever)传输超文本到客户端(本地浏览器…...

访问学者进入美国哪些东西不能带?
随着疫情的稳定,各国签证的逐步放开,成功申请到国外访问学者、博士后如何顺利的进入国外,哪些东西不能带,下面就随知识人网小编一起看一看。一、畜禽肉类(Meats, Livestock and Poultry)不论是新鲜的、干燥的、罐头的、真空包装的…...

灵巧手抓持<分类><仿真>
获取灵巧手抓取物体时的抓持类型,需要考虑:手本身的结构、被抓取物体的形状尺寸、抓持操作任务的条件。 研究方法:基于模型的方法、基于数据驱动的方法 基于模型的方法:建立灵巧手抓持相关的运动学和动力学模型建立目标函数求解…...

CENTO OS上的网络安全工具(十九)ClickHouse集群部署
一、VMware上集群部署ClickHouse (一)网络设置 1. 通过修改文件设置网络参数 (1)CentOS 在CENTOS上的网络安全工具(十六)容器特色的Linux操作_lhyzws的博客-CSDN博客中我们提到过可以使用更改配置文件的方式…...

tesseract -图像识别
下载链接:https://digi.bib.uni-mannheim.de/tesseract/如下选择最新的版本,这里我选择tesseract-ocr-w64-setup-5.3.0.20221222.exe有如下python模块操作tesseractpyocr 国内源:pip install -i https://pypi.mirrors.ustc.edu.cn/simple/ py…...

JavaScript Math 算数对象
文章目录JavaScript Math 算数对象Math 对象Math 对象属性Math 对象方法算数值算数方法JavaScript Math 算数对象 Math(算数)对象的作用是:执行常见的算数任务。 Math 对象 Math(算数)对象的作用是:执行普…...

一体机HDATA节点添加和删除
瀚高数据库 目录 文档用途 详细信息 文档用途 一体机可在线添加、删除数据库集群节点。 详细信息 一体机可在线添加、删除数据库集群节点。具体操作步骤如下 一、节点添加 集群可以在其他机器上通过配置hghac.yaml文件,将新节点加入集群。 集群操作 1…...

关于 interface{} 会有啥注意事项?上
学习 golang ,对于 interface{} 接口类型,我们一定绕不过,咱们一起来看看 使用 interface{} 的时候,都有哪些注意事项吧 interface {} 可以用于模拟多态 xdm 咱们写一个简单的例子,就举动物的例子 写一个 Animal 的…...

Matlab中旧版modem.qammod与新版不兼容
最近,因为课题需要,在研究通信。在网上下了一个2015年左右的代码,其中用的是matlab旧版中的modem.qammod函数,但是旧版中的函数已经被删除了,(这里必须得吐槽一下,直接该函数内部运行机制就行呀…...

通达信指标公式颜色代码的四种写法(COLOR/RGB)
通达信指标公式颜色代码有四种写法,分别为COLOR颜色的英文、COLOR十六进制、RGBX十六进制、RGB(R,G,B)。标题有点尴尬,让我想到孔乙己“茴”字的四种写法,哈哈。 一、COLOR颜色的英文 “COLOR颜色的英文”这种写法比较简单,函数库…...

小程序面试题收集(持续更新中...)
小程序面试题收集 1.请谈谈微信小程序主要目录和文件的作用 project.config.json:项目配置文件,用的最多的就是配置是否开启https校验App.js:设置一些全局的基础数据等App.json:底部tab,标题栏和路由等设置App.wxss&…...

最深情的告白——郁金香(Python实现)
目录 1 最深情的告白 2 即兴赞之 2.1 李小白言郁金香 2.2 郁金香般的姑娘 2.3 荷兰的郁金香 3 Python代码实现 3.1 郁金香的芬芳 3.2 我俩绚丽多姿的风景 1 最深情的告白 曾经以为,她爱玫瑰,然后我画了好几种: 花仙子——玫瑰&a…...

代码随想录算法训练营第六天|242.有效的字母异位词 、349. 两个数组的交集 、 202. 快乐数、1. 两数之和
当我们遇到了要快速判断一个元素是否出现集合里的时候,就要考虑哈希法。哈希法是牺牲了空间换取了时间,要使用额外的数组,set或者是map来存放数据,才能实现快速的查找。当我们要使用集合来解决哈希问题的时候,优先使用…...

【STL】模拟实现list
目录 1、list介绍 所要实现类及其成员函数接口总览 2、结点类的模拟实现 基本框架 构造函数 3、迭代器类的模拟实现 迭代器类存在的意义 3.1、正向迭代器 基本框架 默认成员函数 构造函数 运算符重载 --运算符重载 !运算符重载 运算符重载 *运算符重载 …...

Spring Cloud Alibaba全家桶(五)——微服务组件Nacos配置中心
前言 本文小新为大家带来 微服务组件Nacos配置中心 相关知识,具体内容包括Nacos Config快速开始指引,搭建nacos-config服务,Config相关配置,配置的优先级,RefreshScope注解等进行详尽介绍~ 不积跬步,无以至…...

【微信小程序】-- 页面事件 - 下拉刷新(二十五)
💌 所属专栏:【微信小程序开发教程】 😀 作 者:我是夜阑的狗🐶 🚀 个人简介:一个正在努力学技术的CV工程师,专注基础和实战分享 ,欢迎咨询! &…...

springboot启动过程加载数据笔记(springboot3)
SpringApplication AbstractApplicationContext PostProcessorRegistrationDelegate ConfigurationClassPostProcessor ConfigurationClassParser 一堆循环和调用 ComponentScanAnnotationParser扫描 processConfigurationClass.doProcessConfigurationClass(configClass, so…...

中文代码86
PK 嘚釦 docProps/PK 嘚釦諿A眎 { docProps/app.xml漅薾?糤?D?v拢W4揣狤"攃e9 睔貣m*:PAz韒g?项弇}R珁湧4嶱 ]I禑菦?櫮戵\U佳 珩 ]铒e礎??X(7弅锿?jl筀儸偛佣??z窊梈ZT炰攷 ?\ 銒沆?状尧绥>蕮 ?斬殕{do]?o乗YX?:??罢秗,泿)怟 …...

网络参考模型
OSI参考模型 应用层 不服务于任何其他层,就是位APP提供相应的服务,不如HTTP、域名解析DNS提供服务表示层 1.使得应用数据能够被不同的系统(Windows\Linux)进行识别和理解 2.数据的解码和编码、数据的加密与解密、数据的压缩和解…...

Spark Tungsten
Spark Tungsten数据结构Unsafe Row内存页管理全阶段代码生成火山迭代模型WSCG运行时动态生成Tungsten (钨丝计划) : 围绕内核引擎的改进: 数据结构设计全阶段代码生成(WSCG,Whole Stage Code Generation) 数据结构 Tungsten 在…...

2023年总结的web前端学习路线分享(学习导读)
如果你打开了这篇文章,说明你是有兴趣想了解前端的这个行业的,以下是博主2023年总结的一些web前端的学习分享路线,如果你也想从事前端或者有这方面的想法的,请接着往下看! 前端发展前景 前端入门 巩固基础 前端工程…...

MyBatis学习笔记(十) —— 动态SQL
10、动态SQL MyBatis框架的动态SQL技术是一种根据特定条件动态拼装SQL语句的功能,它存在的意义是为了解决拼接SQL语句字符串的痛点问题。 动态SQL: 1、if 标签:通过test属性中的表达式判断标签中的内容是否有效(是否会拼接到sql中…...

剑指 Offer 55 - II. 平衡二叉树
剑指 Offer 55 - II. 平衡二叉树 难度:easy\color{Green}{easy}easy 题目描述 输入一棵二叉树的根节点,判断该树是不是平衡二叉树。如果某二叉树中任意节点的左右子树的深度相差不超过1,那么它就是一棵平衡二叉树。 示例 1: 给定二叉树 […...

一文吃透前端低代码的 “神仙生活”
今天来说说前端低代码有多幸福? 低代码是啥?顾名思义少写代码…… 这种情况下带来的幸福有:代码写得少,bug也就越少(所谓“少做少错”),因此开发环节的两大支柱性工作“赶需求”和“修bug”就…...

【深度学习】预训练语言模型-BERT
1.BERT简介 BERT是一种预训练语言模型(pre-trained language model, PLM),其全称是Bidirectional Encoder Representations from Transformers。下面从语言模型和预训练开始展开对预训练语言模型BERT的介绍。 1-1 语言模型 语言模型 …...

C++类的组合
C类的组合什么是类的组合初始化参数列表使用类的组合案例分析组合构造和析构顺序问题this指针基本用法和作用什么是类的组合 类的组合就是以另一个对象为数据成员,这种情况称为类的组合 1.优先使用类的组合,而不是继承 2.组合表达式的含义 一部分关系 初…...

2.伪随机数生成器(ctr_drbg)的配置与使用
零、随机数应用 生成盐,用于基于口令的密码 生成密钥,用于加密和认证 生成一次性整数Nonce,防止重放攻击 生成初始化向量IV 构成 种子,真随机数生成器的种子来源于物理现象 内部状态,种子用来初始化内部状态 一、真随机数和伪随机数 1.区别 随机数在安全技术中通常被用于…...

CentOS7 切换图形模式和多用户命令行模式
备注: 主机名 hw 含义:hardware 缩写,意思是硬件(物理机) 文章目录1、查看源头2、查看当前系统运行模式3、设置系统运行模式为多用户命令行模式4、查看当前系统运行模式5、重启系统6、确认当前系统运行模式7、设置系统…...

在linux上用SDKMan对Java进行多版本管理
在linux上用SDKMan对Java进行多版本管理 有一个工具叫SDKMan,它允许我们这样做。官方网站这样描述: TIP: "SDKMan 是一个工具,用于在大多数基于Unix的系统上管理多个软件开发工具包的并行版本。它提供了一个方便的命令行接口(CLI)和API,…...

JSONObject、fastJson(JsonObject)、Gson(JsonObject)区别
概述 Java中并没有内置的 JSON 解析,需要使用第三方类库 fastJson :阿里巴巴的JSON 库,优势在于解析速度快,解析效率高,可以轻松处理大量的 JSON 数据JackSon : 社区十分活跃,spring框架默认使…...