文档章节

HTTP/2--HPACK算法

Candy_Desire
 Candy_Desire
发布于 2015/12/30 16:30
字数 1205
阅读 194
收藏 2

 为 HTTP/2 头压缩专门设计的 HPACK

      HTTP/2 对消息头采用 HPACK 进行压缩传输,能够节省消息头占用的网络的流量。如何理解 HPACK 压缩呢? 如果我们约定将常用的请求头的参数用一些特殊的编号来表示,比如 GET /index.html 用一个 1 来表示,POST /index.html 用 2 来表示。那么是不是可以节省很多字节? 为 HTTP/2 的专门量身打造的 HPACK 便是类似这样的思路延伸。它使用一份索引表来定义常用的 HTTP Header。把常用的 HTTP Header 存放在表里。请求的时候便只需要发送在表里的索引位置即可。例如 :method=GET 使用索引值 2 表示,:path=/index.html 使用索引值 5 表示。 具体这里的静态映射表请看附录的Table 1: Static Table Entries。

 

      比如我们想表达 请求的Header内容是:   GET /index.html   只要给服务端发送一个 Frame,该 Frame 的 Payload 部分存储 0x8285,Frame 的 Type 设置为 Header 类型。

      为什么是 0x8285,而不是 0x0205? 这是因为高位设置为 1 表示这个字节是一个完全索引值(key 和 value 都在索引中)。类似的,通过高位的标志位可以区分出这个字节是属于一个完全索引值,还是仅索引了 key,还是 key 和 value 都没有索引。因为索引表的大小的是有限的,它仅保存了一些常用的 HTTP Header,同时每次请求还可以在表的末尾动态追加新的 HTTP Header 缓存。动态部分称之为 Dynamic Table。Static Table 和 Dynamic Table 在一起组合成了索引表:

参考: http://io.upyun.com/2015/05/13/http2/ 

技术分享

HPACK 不仅仅通过索引键值对来降低数据量,同时还会将字符串进行霍夫曼编码来压缩字符串大小。

以常用的 User-Agent 为例,它在静态表中的索引值是 58,它的值是不存在表中的,因为它的值是多变的。第一次请求的时候它的 key 用 58 表示,表示这是一个 User-Agent ,它的值部分会进行霍夫曼编码(如果编码后的字符串变更长了,则不采用霍夫曼编码)。服务端收到请求后,会将这个 User-Agent 添加到 Dynamic Table 缓存起来,分配一个新的索引值。客户端下一次请求时,假设上次请求User-Agent的在表中的索引位置是 62, 此时只需要发送 0xBE(同样的,高位置 1),便可以代表: User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/33.0.1750.146 Safari/537.36。其过程如下图所示:

 

HPACK: Header Compression for HTTP/2

技术分享

图来自: http://chimera.labs.oreilly.com/books/1230000000545/ch12.html#HTTP2_PUSH

 

WireShark  可以看到具体的 Header 解密后的内容,如下图:

技术分享

上图是用的 WireShark  Development Release (1.99.6) 看到的 Header frame 的内容。

https://www.wireshark.org/download.html 

 

Header  Frame 的数据格式

HTTP2 的 Header Frame 的数据区, 分下面几种格式,相关资料请参考:http://http2.github.io/http2-spec/compression.html#index.address.space

 

1、在 预定的头字段静态映射表 中已经有预定义的 Header Name 和 Header Value值

技术分享

这时候的二进制数据格式如上图, 第一位固定为1, 后面7位为映射的索引值

An indexed header field starts with the ‘1‘ 1-bit pattern, followed by the index of the matching header field, represented as an integer with a 7-bit prefix

例子,

下图的 83 就是这样的,  83 的二进制字节标示  1000 0011 , 抹掉首位, 就是 3 , 对应的 静态映射表中 的 method: POST。

技术分享 

注意,不存在首位为0,但是后面可以在静态表中可以找到的情况。

 

2、 预定的头字段静态映射表中有 name,需要设置新值。

Literal Header Field with Incremental Indexing

这种情况下的数据格式为:

技术分享

例子, 一个指定 path的Header, 

首字符 为 44 ,对应的二进制位   0100 0100   前两个字符为 01 ,  Index 为 4 , 即对应静态映射表中的 path 头。

第二个字符为 95  对应的二进制位 1001 0101 ,排除首字符对应的 Value Length 为 十进制的21。 即 算上 44, 一共23个字符来记录这个信息。

技术分享

 

 

3、预定的头字段静态映射表中没有 name,需要设置新name和新值。

技术分享

例子

40 的二进制是  0100 0000,   
02 的二进制是  0000 0010, 后七位的十进制值是 2

86 的二进制是  1000 0110  后7位的十进制值是 6

技术分享

 

其他还有2种情况:

Literal Header Field without Indexing

  • Literal Header Field without Indexing — Indexed Name

  • Literal Header Field without Indexing — New Name

Literal Header Field Never Indexed


    • Literal Header Field Never Indexed — Indexed Name

    • Literal Header Field Never Indexed — New Name

请参考: http://http2.github.io/http2-spec/compression.html#index.address.space 

 

附录

HTTP2 中, 预定的头字段静态映射表如下。(Table 1: Static Table Entries)

技术分享

 


© 著作权归作者所有

共有 人打赏支持
Candy_Desire
粉丝 31
博文 71
码字总数 84592
作品 0
浦东
产品经理
私信 提问
HPACK 完全解析

去年五月, IETF 正式发布了 HTTP/2 协议与之配套的 HPACK 头部压缩算法。 RFC 如下: Hypertext Transfer Protocol Version 2 RFC 7540 HPACK: Header Compression for HTTP/2 RFC 7541 笔者......

任我行
2016/08/20
0
0
Http2, HTTP1.1 知识点收集

参考地址: https://blog.wangriyu.wang/2018/05-HTTP2.html https://www.jianshu.com/p/e57ca4fec26f 1, HTTP/1.1 存在的问题 TCP 连接数限制 线头阻塞 (Head Of Line Blocking) 问题 Head......

Carlyle_Lee
10/25
0
0
HTTPS、SPDY、HTTP/2 对比

整理自:A Simple Performance Comparison of HTTPS, SPDY and HTTP/2 Firefox 35,成为第一个默认开启支持HTTP/2协议的浏览器。Chrome也支持了,只是以SPDY 4的名义,并且要自己在里面手动开...

Candy_Desire
2015/12/30
263
0
okhttp 3.4.0 和 3.4.1 发布,HTTP+SPDY 客户端开发包

okhttp 3.4.0 和 3.4.1 发布了,okhttp 是一个 Java 的 HTTP+SPDY 客户端开发包,同时也支持 Android。 3.4.0 更新如下: New: Support dynamic table size changes to HPACK Encoder. Fix:......

NextLife
2016/07/11
2.5K
7
HTTP/2 正式通过 IETF 组织批准发布

超过两年的讨论,超过 200 个设计问题, 17 个草案 和 30 个实现, HTTP/2 和 HPACK 规范终于通过了 IETF 组织的批准并作为可跟踪的 RFC 标准。HTTP/2 将帮助实现更快的用户浏览体验,降低带宽...

oschina
2015/02/20
3.3K
6

没有更多内容

加载失败,请刷新页面

加载更多

Confluence 6 快捷键

快捷键图标。 官方的下载地址为:https://atlassianblog.wpengine.com/wp-content/uploads/2018/01/keyboard-shortcuts-infographics.pdf...

honeymose
今天
2
0
Apache限定目录解析PHP,限制user_agent,PHP相关的配置

Apache限定目录解析PHP 配置前访问upload/index.php [root@test-a ~]# curl -x192.168.77.139:80 'www.test.com/upload/index.php'This is upload diretory 配置,/usr/local/apache2.4/......

野雪球
今天
3
0
java.util.Concurrent.Exchanger源码

类图 源码: package java.util.concurrent;import java.util.concurrent.atomic.AtomicInteger;import java.util.concurrent.atomic.AtomicReference;import java.util.concurrent......

狼王黄师傅
今天
6
0
Kubernetes里的secret最基本的用法

Secret解决了密码、token、密钥等敏感数据的配置问题,使用Secret可以避免把这些敏感数据以明文的形式暴露到镜像或者Pod Spec中。 Secret可以以Volume或者环境变量的方式使用。 使用如下命令...

JerryWang_SAP
昨天
5
0
2018-11-20学习笔记

1. python数据类型: 给变量赋值什么样的值,变量就是什么样的类型 给变量赋值整数,变量就是整数类型 给变量赋值字符串,变量就是字符串类型 123 和“123”一样吗? 在python中 单引号 与双...

laoba
昨天
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部