探索HTTP/2

文章目录

    • http/1.1
    • http/2
      • 疑惑
    • 探索
      • 1. 连接前言
      • 2. 帧结构
        • 2.1 帧类型 Type
      • 3. 帧详情
        • 3.1 SETTINGS 帧
        • 3.2 WINDOW_UPDATE 帧
        • 3.3 PRIORITY 帧
        • 3.4 HEADERS 帧
        • 3.5 DATA 帧
        • 3.6 PING
        • 3.7 GOAWAY 帧
        • 3.8 RST_STREAM 帧
        • 3.9 PUSH_PROMISE 帧
        • 3.10 CONTINUATION 帧

在这里插入图片描述

你对http2了解多少? 你愿意在你的项目中使用http2吗?为什么?

http/1.1

在这里插入图片描述

从上图可以看到,是一个人在银行排队等待办理一个业务,完成后再去排队办理下一个业务。这就是http/1.1的请求过程。

http/1.1是串行的,也就是说,每个请求都是按顺序进行的,如果前面的请求没有完成,后面的请求就必须等待。

我们无法知道确定之前的顾客在干嘛,可能是在办理业务,也可能是在等待,也可能是在疯狂的聊天直到世界末日。

http/2

在这里插入图片描述

从上图可以看到,是一个人在银行办理多个业务,每个业务都是同时进行的。这就是http/2的请求过程。

http/2是并行的,也就是说,每个请求都是同时进行的,不需要等待前面的请求完成。

疑惑

这样高强度的并行请求,会不会导致服务器压力过大?会不会导致服务器崩溃?

其实不会,http/1.1 并没有榨干TCP协议所能提供的性能。

探索

了解http/2的中的一些细节。

1. 连接前言

在 http/2 中,连接成功后,每个端点都需要发送连接序言作为所使用协议的最终确认。

  • 客户端发送 PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n 作为连接序言
  • 服务器端回应 SETTINGS 帧,帧的内容是服务器端的初始设置,包括最大帧大小,最大并发流数等。

需要注意的是,为了避免不必要的延迟,客户端并不会等待服务器端的回应,而是直接发送请求。

2. 帧结构

在这里插入图片描述

名称长度描述
Length3 字节帧负载的长度(取值范围为 214~224-1 字节),使用SETTINGS 帧可以设置更大值
Type1 字节帧类型
Flags1 字节特定于帧类型的标志
R1 位保留位,无需设置
Stream Identifier31 位流标识符
Frame Payload可变长度帧有效负载,长度在 Length 字段中设置
2.1 帧类型 Type
名称ID描述
DATA0x0传输流的核心内容
HEADERS0x1包含 HTTP 首部,和可选的优先级参数
PRIORITY0x2指示或者更改流的优先级和依赖
RST_STREAM0x3允许一端停止流(通常由于错误导致的)
SETTINGS0x4协商连接级参数
PUSH_PROMISE0x5提示客户端,服务器要推送些东西
PING0x6测试连接可用性和往返时延(RTT)
GOAWAY0x7告诉另一端,当前端已结束
WINDOW_UPDATE0x8协商一端将要接收多少字节(用于流量控制)
CONTINUATION0x9用以扩展 HEADER 数据块
更详细的内容可以参考:

  • Hypertext Transfer Protocol Version 2 (HTTP/2)

3. 帧详情

3.1 SETTINGS 帧

在 http/2 中,请求是通过流的方式进行的。

通常,在发送 连接前言 之后,客户端会发送一个 SETTINGS 帧。这个帧包含了客户端的初始设置,包括最大帧大小,最大并发流数等。

  • 帧大小:默认最大值 2^14 字节 (16,384 字节),如果想要更大的帧大小,可以接收方通过 SETTINGS 帧进行设置。最高可以设置为 2^24-1 字节 (16,777,215 字节)。

示例(十六进制):

00 00 12 04 00 00 00 00 00 00 01 00 01 00 00 00 04 00 02 00 00 00 05 00 00 40 00

这个帧的内容是:

  • 帧长度:18
  • 帧类型:4 (SETTINGS)
  • 标志位:0
  • 流ID:0
  • 帧有效负载:00 01 00 01 00 00 00 04 00 02 00 00 00 05 00 00 40 00



关于R位,是保留位,值为 0。

SETTINGS 帧的有效负载可以设置多个参数,由 16位 类型和 32位 值组成。

 +-------------------------------+
 |       Identifier (16)         |
 +-------------------------------+-------------------------------+
 |                        Value (32)                             |
 +---------------------------------------------------------------+
类型描述
0x1允许发送方通知远程端点用于解码标头块的标头压缩表的最大大小(以八位位组为单位)。
0x2此设置可用于禁用服务器推送
0x3指示发送方允许的并发流的最大数量
0x4指示发送方用于流级流量控制的初始窗口大小(以八位字节为单位)
0x5指示发送方愿意接收的最大帧有效负载的大小(以八位位组为单位)。
0x6此建议设置通知对等方发送方准备接受的标头列表的最大大小(以八位位组为单位)。
`SETTINGS` 帧的流ID一般为0,表示这是一个连接级的设置,而不是针对某个流的设置。
3.2 WINDOW_UPDATE 帧

在 http/2 中,流量控制是通过 WINDOW_UPDATE 帧来进行的。

WINDOW_UPDATE 帧的有效负载是一个 1 位的 R 位,和 31 位的 增加窗口大小

 +-+-------------------------------------------------------------+
 |R|              Window Size Increment (31)                     |
 +-+-------------------------------------------------------------+

帧类型是 0x8,表示这是一个 WINDOW_UPDATE 帧。

帧标志位有两种,分别是 流流控窗口和连接流控窗口,如果是流流控窗口,则指示受影响的流,如果是连接流控窗口,流标识符为 0x0

示例(十六进制):

00 00 04 08 00 00 00 00 00 00 bf 00 01 
3.3 PRIORITY 帧

PRIORITY 帧(帧类型 0x2)用于指示或更改流的优先级和依赖。

使用 `PRIORITY` 帧时,流标识符不能为 `0x0`。

PRIORITY 帧的有效负载是一个 1 位的 E 位,和 31 位的 Stream Dependency,和 8 位的 Weight

 +-+-------------------------------------------------------------+
 |E|                  Stream Dependency (31)                     |
 +-+-------------+-----------------------------------------------+
 |   Weight (8)  |
 +-+-------------+
  • E 位:如果为 0,则当前流的依赖关系被视为非独占的,这意味着当前流与依赖于同一流的其他流共享依赖关系。如果为 1,则当前流的依赖被认为是独占的,当前流将独占其依赖流,所有其他原本依赖于那个流的流将改为依赖当前流。
  • Stream Dependency:指示当前流的依赖流的标识符。
  • Weight:指示当前流的权重,取值范围为 1~256

示例(十六进制):

00 00 05 02 00 00 00 00 07 00 00 00 00 00
3.4 HEADERS 帧

HEADERS 帧(帧类型 0x1)主要用于传输HTTP请求或响应的头部信息。

HEADERS 帧的有效负载:

类型大小描述
Pad Length8 位用于填充的字节长度(需要有 PADDED 标志)
E1 位指示流依赖性是排他(需要有 Priority 标志)
Stream Dependency31 位该流所依赖的流的 31 位流标识符(需要有 Priority 标志)
Weight8 位用于指示流的权重
Header Block Fragment可变用于存放头部信息
Padding可变用于填充的字节
 +---------------+
 |Pad Length? (8)|
 +-+-------------+-----------------------------------------------+
 |E|                 Stream Dependency? (31)                     |
 +-+-------------+-----------------------------------------------+
 |  Weight? (8)  |
 +-+-------------+-----------------------------------------------+
 |                   Header Block Fragment (*)                 ...
 +---------------------------------------------------------------+
 |                           Padding (*)                       ...
 +---------------------------------------------------------------+

HEADERS 帧标志:

- `END_STREAM (0x1)`:指示这是流的最后一个帧。
- `END_HEADERS (0x4)`:指示这是头部块的最后一个帧。
- `PADDED (0x8)`:指示这个帧包含了填充字段。
- `PRIORITY (0x20)`:指示这个帧包含了优先级信息。
了解这个帧的结构,可以更好的理解http/2的请求过程。这里可以查看hpack压缩算法,了解更多关于头部块的内容。

https://datatracker.ietf.org/doc/html/rfc7541

3.5 DATA 帧

DATA 帧(帧类型 0x0)用于传输流的核心内容。

+-------------+ 
 |焊盘长度?(8)| 
 +------------------------++--------------------------------- --------------+ 
 | 数据 (*) ... 
 +---------------------------------------- ------------------+ 
 | 填充 (*) ... 
 +---------------------------------------- ------------------+

DATA 帧的有效负载:

  • Pad Length:用于填充的字节长度(需要有 PADDED 标志)。
  • Data:用于存放数据。数据量是帧有效负载减去存在的其他字段的长度后的剩余部分。
  • Padding:用于填充的字节。(填充设置为 0 )

DATA 帧标志:

  • END_STREAM (0x1):指示这是流的最后一个帧。
  • PADDED (0x8):指示这个帧包含了填充字段。

示例(十六进制):

00 00 0c 00 01 00 00 00 0f 7b 22 64 61 79 4e 6f 22 3a 33 30 7d     
3.6 PING

PING 帧(帧类型 0x6)用于测试连接的可用性和往返时延(RTT)。

 +---------------------------------------------------------------+
 |                                                               |
 |                      Opaque Data (64)                         |
 |                                                               |
 +---------------------------------------------------------------+

PING 帧的有效负载:

  • Opaque Data:用于存放数据。发送者可以包含它选择的任何值并以任何方式使用这些八位字节。

PING 帧标志:

  • ACK (0x1):指示该 PING 帧是 PING 响应。端点必须在 PING 响应中设置此标志。端点不得响应包含此标志的 PING 帧。
流标识符为 `0x0`,表示这是一个连接级的PING帧。否则,视为连接错误。

示例(十六进制):

00 00 08 06 00 00 00 00 00 00 00 00 00 00 00 00 00
3.7 GOAWAY 帧

GOAWAY 帧(帧类型 0x7)用于告诉另一端,当前端已结束。

 +-+-------------------------------------------------------------+
 |R|                  Last-Stream-ID (31)                        |
 +-+-------------------------------------------------------------+
 |                      Error Code (32)                          |
 +---------------------------------------------------------------+
 |                  Additional Debug Data (*)                    |
 +---------------------------------------------------------------+

GOAWAY 帧的有效负载:

  • R:保留位,值为 0
  • Last-Stream-ID:指示发送端认为的最后一个流的标识符。
  • Error Code:用于指示连接关闭的原因。
  • Additional Debug Data:用于存放额外的调试数据。

GOAWAY 帧行为:

  • 发送了GOAWAY帧的一端将忽略所有在GOAWAY帧指定的最后流标识符之后启动的流的帧。
  • 接收GOAWAY帧的一端不应再在该连接上开启新的流,但可以建立新的连接来发起更多的流。
  • 如果接收方在流标识符比GOAWAY帧中指示的更高的流上发送了数据,这些流将不会被处理。接收方可以认为这些流就像从未被创建过一样,并可以在新的连接上重试它们。

示例(十六进制):

00 00 08 07 00 00 00 00 00 00 00 00 00 00 00 00 00
3.8 RST_STREAM 帧

RST_STREAM 帧(帧类型 0x3)用于允许一端停止流(通常由于错误导致的)。

 +---------------------------------------------------------------+
 |                        Error Code (32)                        |
 +---------------------------------------------------------------+

RST_STREAM 帧的有效负载:

  • Error Code:用于指示流关闭的原因。
和`GOAWAY`相比,`GOAWAY`针对整个连接,而 `RST_STREAM` 仅针对单个流。 `RST_STREAM` 帧的流标识符不能为 `0x0`。

示例(十六进制):

00 00 04 03 00 00 00 00 03 00 00 00 00
3.9 PUSH_PROMISE 帧

PUSH_PROMISE 帧(帧类型 0x5)用于提前通知对等端点发送方打算发起的流。PUSH_PROMISE 帧包括端点计划创建的流的无符号 31 位标识符以及为流提供附加上下文的一组标头。

 +---------------+
 |Pad Length? (8)|
 +-+-------------+-----------------------------------------------+
 |R|                  Promised Stream ID (31)                     |
 +-+-------------+-----------------------------------------------+
 |                   Header Block Fragment (*)                 ...
 +---------------------------------------------------------------+
 |                           Padding (*)                       ...
 +---------------------------------------------------------------+

PUSH_PROMISE 帧的有效负载:

  • Pad Length:用于填充的字节长度(需要有 PADDED 标志)。
  • R:保留位,值为 0
  • Promised Stream ID:指示发送方打算创建的流的标识符。
  • Header Block Fragment:用于存放头部信息。
  • Padding:用于填充的字节。

PUSH_PROMISE 帧标志:

- `END_HEADERS (0x4)`:指示这是头部块的最后一个帧。
- `PADDED (0x8)`:指示这个帧包含了填充字段。

示例(十六进制):

00 00 04 05 04 00 00 00 00 00 00 00 01

PUSH_PROMISE帧允许服务器主动向客户端推送资源,而不需要客户端显式地请求这些资源。这是一种服务器推送技术,旨在减少网页加载时间。通过预先发送客户端将需要的资源,服务器可以显著减少网页完成渲染所需的往返次数(RTTs)。

但是,由于服务器推送可能会导致客户端缓存不一致,因此,服务器推送的资源应该是客户端已经请求过的资源,或者是客户端可能会请求的资源。

3.10 CONTINUATION 帧

CONTINUATION 帧(帧类型 0x9)用于扩展 HEADER 数据块。

 +---------------------------------------------------------------+
 |                   Header Block Fragment (*)                 ...
 +---------------------------------------------------------------+

CONTINUATION 帧的有效负载:

  • Header Block Fragment:用于存放头部信息。

CONTINUATION 帧标志:

  • END_HEADERS (0x4):指示这是头部块的最后一个帧。(如果未设置 END_HEADERS 位,则该帧后面必须跟有另一个 CONTINUATION 帧。)

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

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

相关文章

BUUCTF--极客大挑战php

文章目录 1.网站备份文件www.zip2.下载后发现class.phpindex.phpflag.php 3.分析php代码绕过__wakeup方法变量权限为私有或保护python方法url方法 1.网站备份文件www.zip 2.下载后发现 class.php <?php include flag.php; error_reporting(0);class Name{private $usernam…

MYSQL5.7报1205 - Lock wait timeout exceeded; try restarting transaction

简介 今天使用navicate操作添加时&#xff0c;mysql报错误&#xff0c;错误如下 原因 这个问题的原因是在mysql中产生了事务A&#xff0c;执行了修改的语句&#xff0c;比如&#xff1a; update t1 set aget18 where id1;此时事务并未进行提交&#xff0c;事务B开始运行&am…

【C++精简版回顾】19.异常处理

1.throw抛出问题 int print(int a,int b) {if (b 0)throw b;return a / b; } 2.try与catch解决问题 try {print(2, 0); } catch (int b) {cout << "竟然是&#xff1a;"<<b<<endl; } 结果&#xff1a; 补充1&#xff1a;可以抛出字符串等 1.throw…

pytorch配置环境

1.查看cuda版本 nvidia-smi cuda version:12.3 2.下载torch 然后根据版本去查找对应的 torch下载代码 可查看这里&#xff1a;Previous PyTorch Versions | PyTorch 然后执行 conda install pytorch1.10.0 torchvision0.11.0 torchaudio0.10.0 cudatoolkit11.3 -c pytorch…

Python的文件操作

归纳编程学习的感悟&#xff0c; 记录奋斗路上的点滴&#xff0c; 希望能帮到一样刻苦的你&#xff01; 如有不足欢迎指正&#xff01; 共同学习交流&#xff01; &#x1f30e;欢迎各位→点赞 &#x1f44d; 收藏⭐ 留言​&#x1f4dd; 路在脚下&#xff0c;勇往直前&#x…

使用nvidia-ml-py事实监控GPU状态

平时监控GPU状态最常用的是watch配合nvidia-smi指令&#xff0c;但有时可能不仅仅需要监控&#xff0c;还需要记录状态数据&#xff0c;比如GPU的显存变化以及利用率变化等等。本文提供了一个使用nvidia-ml-py包编写的简易Demo&#xff0c;该Demo能够实现简易版的nvidia-smi功能…

[云原生] K8s之pod控制器详解

Pod 是 Kubernetes 集群中能够被创建和管理的最小部署单元。所以需要有工具去操作和管理它们的生命周期,这里就需要用到控制器了。 Pod 控制器由 master 的 kube-controller-manager 组件提供&#xff0c;常见的此类控制器有 Replication Controller、ReplicaSet、Deployment、…

openssl3.2 - exp - 产生随机数

文章目录 openssl3.2 - exp - 产生随机数概述笔记END openssl3.2 - exp - 产生随机数 概述 要用到openssl产生的随机数, 查了资料. 如果用命令行产生随机数, 如下: openssl rand -hex -num 6 48bfd3a64f54单步跟进去, 看到主要就是调用了一个RAND_bytes(), 没其他了. 官方说…

美国站群服务器使用技巧与注意事项

美国站群服务器使用技巧与注意事项有哪些?RAKsmart小编为您整理发布美国站群服务器使用技巧与注意事项&#xff0c;希望对您有帮助。 美国站群服务器的使用技巧主要包括远程管理、灵活配置和备份还原&#xff0c;具体如下&#xff1a; 1. **远程管理**&#xff1a;用户可以通过…

Promisification、微任务

前提摘要 Promise 对象的构造器&#xff08;constructor&#xff09;语法如下&#xff1a; let promise new Promise(function(resolve, reject) { // executor }); 传递给 new Promise的函数被称为 executor&#xff0c;当 new Promise 被创建&#xff0c;executor 会自动…

本地部署websocket服务端并结合内网穿透实现固定公网地址连接

文章目录 1. Java 服务端demo环境2. 在pom文件引入第三包封装的netty框架maven坐标3. 创建服务端,以接口模式调用,方便外部调用4. 启动服务,出现以下信息表示启动成功,暴露端口默认99995. 创建隧道映射内网端口6. 查看状态->在线隧道,复制所创建隧道的公网地址加端口号7. 以…

So you think you understand IP fragmentation?

文章目录 前言一、Why care?二、Prevention三、Well-understood?四、Introducing fragquiz五、A novel (?) algorithm六、Reader challenge七、traceroute八、ICMP参考资料 前言 本文来自&#xff1a;https://lwn.net/Articles/960913/ February 7, 2024This article was …

英福康INFICON真空计VGC012-103-401使用说明

英福康INFICON真空计VGC012-103-401使用说明

【C++ Primer Plus学习记录】break和continue语句

break和continue语句都使程序能够跳过部分代码。可以在switch语句或任何循环中使用break语句&#xff0c;使程序跳到switch或循环后面的语句处执行。continue语句用于循环中&#xff0c;让程序跳过循环体中余下的代码&#xff0c;并开始新一轮循环&#xff08;参见图6.4&#x…

Unity性能优化篇(八) 导入的模型网格优化设置

模型导入Unity后&#xff0c;可以选中这个模型&#xff0c;在Inspector窗口设置它的属性。下面说的都是可自定义选择优化的地方 Model选择卡: 1.在Model选项卡&#xff0c;启用Mesh Compression可以压缩模型&#xff0c;压缩程度越高&#xff0c;模型精度越低&#xff0c;但是…

GEE 将裁剪后的Sentinel-2影像 在ArcGIS中去除空值

在ArcGIS中&#xff0c;可以使用栅格计算器&#xff08;Raster Calculator&#xff09;工具来设置NoData值为空。以下是在ArcGIS中将NoData值设置为空的步骤&#xff1a; 打开ArcGIS软件并加载下载的Sentinel-2影像数据。 影像Nodata空值以黑色背景呈现&#xff0c;影响矢量数据…

使用 Docker 部署 Stirling-PDF 多功能 PDF 工具

1&#xff09;Stirling-PDF 介绍 大家应该都有过这样的经历&#xff0c;面对一堆 PDF 文档&#xff0c;或者需要合并几个 PDF&#xff0c;或者需要将一份 PDF 文件拆分&#xff0c;又或者需要调整 PDF 中的页面顺序&#xff0c;找到的线上工具 要么广告满天飞&#xff0c;要么 …

OpenCV与机器学习:OpenCV实现主成分分析

OpenCV实现主成分分析 前言主成分分析&#xff08;PCA&#xff09;数据生成画图cv2.PCACompute绘制主成分分析结果 前言 维数灾难是指出现一定数量的特征&#xff08;或者维度&#xff09;后&#xff0c;分类器的性能将开始下降。特征越多&#xff0c;数据集中的信息就越多。但…

当前爆火的:ChatGPT4、Claude3、Gemini、Sora、GPTs及AI领域中的集中大模型的最新技术

原文链接&#xff1a;当前爆火的&#xff1a;ChatGPT4、Claude3、Gemini、Sora、GPTs及AI领域中的集中大模型的最新技术 第一&#xff1a;2024年AI领域最新技术 1.最新超强模型Claude3使用 2.OpenAI新模型-GPT-5 3.谷歌新模型-Gemini使用 4.Meta新模型-LLama3 5.阿里巴巴…

[C语言]——C语言常见概念(1)

目录 一.C语言是什么、 二.C语言的历史和辉煌 三.编译器的选择&#xff08;VS2022为例&#xff09; 1.编译和链接 2.编译器的对比 3.VS2022 的优缺点 四.VS项目和源文件、头文件介绍 五.第⼀个C语言程序 ​​​​​​​ 一.C语言是什么、 ⼈和⼈交流使⽤的是⾃然语⾔&…