五千多字,图文并茂详解HTTP报文格式、请求响应头、cookie以及HTTPS加密方式

  • 靓仔靓女们大家好,我们又见面了,:java小杰要加油,这周来分享一篇关于HTTP协议相关的文章
  • 看完此文可以对
    • HTTP报文格式HTTP各种请求头HTTP响应码cookie属性以及HTTPS为什么安全(涉及到三种加密方式) 有个清晰的认知
  • 全文五千来字,强烈建议收藏,巩固基础
  • 若文中涉及到的知识点有所偏差的话,还请大佬们指出,小杰感激不尽,冲冲冲!~
  • 话不多说,直接开搞

HTTP简介

  • 超文本传输协议(Hypertext Transfer Protocol,HTTP)是一个简单的请求-响应协议,它通常运行在TCP之上。它指定了客户端可能发送给服务器什么样的消息以及得到什么样的响应
  • 现在主要应用 http1.1 协议
  • http是无状态协议,不会保存多次请求之间的关系,使用cookie做状态管理
  • 持久连接节省通信量(HTTP1.1和部分HTTP1.0)
  • 通过请求方法告知服务器意图,get,post

HTTP报文

  • 用于HTTP协议交互的信息叫做HTTP报文
  • 报文由报文首部报文主体来组成,其中由空行分割
  • 请求报文响应报文报文结构不一样,其中最大的区别就是在报文首部中,各有各的特定的首部
image
  • 报文首部:服务器或者客户端需要处理的请求或者响应的内容及其属性
  • 报文主体:被发送的数据

HTTP请求报文结构

  • 客户端发送的报文叫做请求报文
image
  • 请求行:包含用于请求的方法,请求URI和HTTP版本
  • 请求首部字段:请求报文里特有的字段(后文会提到)
  • 通用首部字段:请求报文和响应报文都会用到的首部
  • 实体首部字段:针对请求报文的实体部分使用的首部
  • 其他:可能包含HTTP的RFC里未定义的首部(如Cookie等)

HTTP响应报文结构

  • 服务端发送的报文叫做响应报文
image
  • 状态行:包含表明响应结果的状态码,原因短语和HTTP版本
  • 响应首部字段:响应报文里特有的字段(后文会提到)
  • 通用首部字段:请求报文和响应报文都会用到的首部
  • 实体首部字段:针对响应报文的实体部分使用的首部
  • 其他:可能包含HTTP的RFC里未定义的首部(如Set-Cookie等)

注:若HTTP首部字段重复了的话,不同的浏览器处理机制不一样

  • 有些浏览器会优先处理第一次出现的字段
  • 有些浏览器会优先处理最后一次出现的字段

HTTP响应码

2xx 成功

2xx的响应结果就代表请求被正常处理了

  • 200 OK:表示客户端发来的请求被服务器正常处理了
  • 204 Not Content:请求被成功处理,但是返回的响应报文不包含实体的主体部分
  • 206 Partial Content:客户端进行范围请求,而服务器重新执行了这部分的GET请求

3xx 重定向

3xx的响应结果就表明浏览器需要执行某些特殊的处理以正确处理请求

  • 301 Moved Permanently:永久重定向。表示请求的资源已经被分配了新的URI,以后应该使用新的URI
  • 302 Found:临时重定向。代表资源只是暂时的移动了以后还可能会移动为新的URI
  • 303 See Other:由于请求对应的资源存在着另一个URI,应使用GET方法定向获取请求的资源
  • 304 Not Modified:客户端发送附带条件(请求首部中if开头的属性中的一种)的请求的时候,服务端允许访问资源,但是那些请求并没有满足,直接返回304,即服务端资源未改变,可以直接使用客户端未过期的缓存,304返回时,不包含任何响应的主体部分(虽然被划分为3xx里,但是和重定向没有任何关系)

4xx 客户端错误

4xx的响应结果就表明客户端是发生错误的原因所在

  • 400 Bad Requset:请求报文中存在语法错误,请修改请求内容后再发送请求
  • 401 Unauthorized:客户端未认证授权
  • 403 Forbidden:服务端禁止客户端访问此资源
  • 404 Not Found:URL写错了,找不到此路径

5xx 服务器错误

5xx的响应结果就表明服务器本身发生错误

  • 500 Internal Server Error:服务器内部故障,可能是bug导致的
  • 503 Service Unavaliable:服务器暂时不可用(停机维护或者超负载),如果事先知道解除这种情况所需的时间,最好写入响应头中的Retry-After这个字段再返回给客户端

HTTP报文首部

HTTP1.1 规定了 以下47种首部字段

通用首部 (共9种)

首部字段 解释
1. Cache-Control 控制缓存的行为
2. Connection 逐跳首部、连接的管理
3. Date 创建报文的日期和时间
4. Pragma 报文指令
5. Trailer 报文末端的首部一览
6. Transfer-Encoding 指定报文主体的传输编码方式
7. Upgrade 升级为其他协议
8. Via 代理服务器的相关信息
9. Warning 错误通知

下面我们来看挑几个重要的属性来看下~

  1. Connection 他有两个作用
  • 控制不再转发给代理的首部字段
GET / HTTP/1.1
Upgrade: HTTP/1.1    // 就会把次字段删除后再从代理服务器转发出去
Connection: Upgrade   // 不再转发的首部字段名
image
  • 管理持久链接(这个比较常见)
    • HTTP/1.1默认连接都是持久连接Connction: Keep-Alive
    • 当服务器想断开的时候,需要指定Connction: close
  1. Pragme :是HTTP/1.1之前版本遗留的字段,仅仅是为了与HTTP/1.0向后兼容而定义
    • Pragm:no-cache :通用首部字段,在请求头中,表示所有的中间服务器不返回缓存的资源
    • 可是所有的中间服务器都以HTTP/1.1为基准的话,可以直接采用 Cache-Control:no-cache
    • 所以一般会发送两个字段Cache-Control:no-cache Pragm:no-cache

请求报文首部 (共19种)

首部字段 解释
1.Accrpt 用户代理可处理的媒体类型
2.Accrpt-Charset 优先的字符集
3.Accept-Encoding 优先的内容编码
4.Accept-Language 优先的语言(自然语言)
5.Authorization web认证信息
6.Expect 期待服务器的特定行为
7.From 用户的电子邮箱地址
8.Host 请求资源所在服务器
9.If-Match 比较实体标记(ETag)
10.If-Modified-Since 比较资源的更新时间
11.If-None-Match 比较实体标记(与If-Match相反)
12.If-Range 资源未更新时发送实体Byte的范围请求
13.If-Unmodified-Since 比较资源的更新时间(与If-Modified-Since相反)
14.Max-Forwards 最大传输逐跳数
15.Proxy-Authorization 代理服务器要求客户端的认证信息
16.Range 实体的字节范围要求
17.Referer 对请求中URI的原始获取方
18.TE 传输编码的优先级
19.User-Agent HTTP客户端程序的信息
  1. If-Match:只有当 If-Match 字段值跟 ETag 值匹配一致时,服务器才会接受请求
    • 它会告知服务器匹配资源所用的实体标记(ETag)值,这时服务器无法使用弱ETag值
    • 仅当两者一致时才会执行请求,否则返回412 Precondition Failed的响应
    • 还可以使用 * 号指定If-Match的字段值,如果这样的话,那么服务器将会忽略ETag的值,只要资源存在就处理请求。
  2. If-Modified-Since
    • 若资源更新时间确实在此字段指定时间之后的话,则处理该请求,否则返回304 Not Modified
    • 用于确认代理或客户端拥有本地资源的有效性,若想获取资源的更新日期时间的话可以通过确认首部字段Last-Modified来确定
  3. If-None-Match
    • 只有在If-None-Match的字段值与ETag值不一致时,才可以处理该请求,与前文中提到的If-Match作用相反
  4. If-Range
    • 他告知服务器若指定的If-Range字段值(ETag值或者时间)和请求资源的ETag值或时间一致时,则作为范围请求处理,否则,返回全体资源
  5. If-Unmodified-Since
    • 指定的请求资源只有在字段值内指定的日期时间之后未发生更新,才会执行这个请求,否则,返回412 Precondition Failed状态响应,与If-Modified-Since作用相反
  6. Max-Forwards
    • 每次请求转发时数值减一,直到0时返回响应
    • 有可能这个请求经过了多台服务器代理转发,如果突然间请求出现了什么问题导致转发失败,而客户端不知道,此时就可以用此属性来定位问题,这个时候我们就可以掌握一个出问题的转发路径,从而方便进一步的排查问题。
  7. Range:
    • 对于只需要获取部分资源的范围请求,Range字段可以指定获取资源范围Range: bytes=10001-20000
    • 例子中表示请求获取从第10001字节到20000字节的资源
    • 服务器处理请求后会返回206 Partial Content的响应。无法处理时,则会返回状态码200 OK的响应及其全部资源

响应报文首部 (共9种)

首部字段名 解释
1.Accept-Ranges 是否接受字节范围请求
2.Age 推算资源创建经过时间
3.ETag 资源的匹配信息
4.Location 令客户端重定向至指定URI
5.Proxy-Authenticate 代理服务器对客户端的认证信息
6.Retry-After 对再次发起请求的时机要求
7.Server HTTP服务器的安装信息
8.Vary 代理服务器缓存的管理信息
9.WWW-Authenticate 服务器对客户端的认证信息
  1. Accept-Ranges
    • Accept-Ranges:bytes 可以处理范围请求
    • Accept-Ranges:none 不可以处理范围请求
  2. Age
    • 可以告知客户端,源服务器多久之前创建了资源,单位是秒
    • 若创建该响应的是缓存服务器,则Age值是指缓存后的响应再次发起发起认证到认证完成的时间值。代理创建响应时必须加上首部字段Age
  3. ETag

    它是一种可将资源以字符串形式做唯一标识的方式,服务器会为每份资源分配对应的ETag值,资源被更新时,ETag值也会被更新,并没有统一的算法规则,而是由服务器来分配

    • ETag:无论实体发生多么细微的变化都会改变其值
    • ETag:只用于提示资源是否相同,只有资源发生了根本的改变才会改变ETag值,这时会在字段值最开始加W/,
      ETag:W/"XXX"
  4. Location
    • 使用该响应字段可以将响应接收方引导至某个与请求的URI位置不同的资源
    • 基本上,该字段配合3XX,Redirection的响应,提供重定向的URI
  5. Vary

    首部字段vary可对缓存进行控制,源服务器会向代理服务器传达关于本地缓存使用方法的命令

    • 当代理服务器接收到服务器返回包含Vary指定项的响应后,仅对请求中含有相同Vary指定首部字段的请求返回缓存
    • 即使对相同资源发起请求,但是由于Vary指定的首部字段不相同,因此必须从源服务器重新获取资源
    • 例如下面这个,如果使用的Accept-Language:en-us字段的值相同,那么直接从缓存返回响应,否则从源服务器请求资源后再返回响应
      image

实体报文首部 (共10种)

首部字段名 解释
1.Allow 资源可支持的HTTP方法
2.Content-Encoding 实体主体适用的编码方式
3.Content-Language 实体主体的自然语言
4.Content-Length 实体主体的大小(单位:字节)
5.Content-Location 代替对应资源的URI
6. Content-MD5 实体主体的报文摘要
7. Content-Range 实体主体的位置范围
8. Content-Type 实体主体的媒体类型
9.Expires 实体主体过期的日期时间
10.Last-Modified 资源的最后修改日期时间

其他字段(cookie等)

  • cookie,我们下面单独讲这个

cookie

  • 注 : 文中例子中的各种请求,报文,均来自 京东物流官网
  • ps:小杰个人挺喜欢JDL的标语的,有速度,更有温度,祝JDL越来越好!
image

set-cookie

属性 解释
NAME=VALUE 赋予Cookie的名称和其值(必需项)
expires = DATE Cookie的有效期(若不指定则默认为浏览器关闭为止)
path=PATH 将服务器上的文件目录作为Cookie的适用对象(若不指定则默认为文档所在的文件目录)
domin=域名 作为Cookie适用对象的域名(若不指定则默认为创建Cookie的服务器域名)
Secure 仅在HTTPS安全通信时才会发送Cookie
HttpOnly 加以限制,使Cookie不能被JS脚本访问,主要目的是为了防止跨站脚本攻击(Cross Site Scripting,XSS)对cookie的窃取

谷歌浏览器控制台查看cookie

image
  • cookie中的thorJSESSIONID这两个key的后HttpOnly属性被打上了√,就表明,此key无法被js脚本访问,防止跨站脚本攻击(Cross Site Scripting,XSS)对cookie的窃取
    我们来看下再console控制台输入document.cookie
    得出的cookie无法找到这两个key
image
image
image

因为这个属性JSESSIONID比较重要,存储的是sessionId,这个要是被别人拿到的话,别人就可以冒充我在网站上做某些事情了,像我自己一样请求某些数据了

postman 模拟拿到cookie后发送请求

image
image

我把网页上的cookie拿下来,放到postman里测试,发现和我自己在网站上请求数据是一样的

cookie存储的地方,清理缓存到底是清理什么?

清理缓存主要就是清理cookie,抹去自己登陆痕迹以及浏览器中的资源缓存,重新请求网站资源

image
image
image

HTTP 与 HTTPS

HTTP不足

  • 通信使用明文(不加密),内容可能会被篡改
  • 不验证通信方的身份,因此有可能遭遇伪装
  • 无法证明报文的完整性,所以有可能已遭遇篡改

HTTPS结构

HTTPS是身披SSL(Secure Socket Layer)外壳的HTTP

image
  • 在采用SSL后,HTP就拥有了HTTPS的加密、证书和完整性保护这些功能
  • 想要了解HTTPS是怎么加密的,得先了解下下面两种提到的加密技术

对称加密原理

  • 客户端和服务端约定好用同一把密钥
  • 这把密钥可以对数据进行加密/解密
image
  • 客户端和服务端之间的共享密钥的传送问题也是一个问题,如果能够安全传送不被截获的话,那岂不是数据也可以安全的传送到不被截获?鸡生蛋蛋生鸡的问题。
  • 图中客户端和服务端传输加密数据的时候,如果双方的共享密钥泄露的被黑客截取到的话,黑客就可以用它来解开这加密的数据,所以对称加密不安全

非对称加密原理(公开密钥加密)

  • 一共有两把密钥(是一对),一个公开密钥,一个私人密钥
  • 公开密钥加密的数据,只有对应的私钥才可以解密,
  • 私钥加密的数据,公钥也可以解密
image
  • 问题就是,从服务端发送给客户端数据时无法保证数据的安全性,因为此时有可能黑客截获到了公钥,对私钥加密的数据进行了解密
  • 服务器端为什么不发送用公钥加密的数据?因为客户端没有私钥,无法解密。

混合加密原理

聪明的大佬们用两种加密算法混合了一下

  • 客户端一开始向服务端传输的是,用公开密钥加密的共享密钥!
  • 这样的话,服务端收到这个加密的数据后用自己的私钥密钥解密后得到的就是共享密钥,以后和客户端交互时都用这个共享密钥就可以啦,因为黑客是无法获得这个共享密钥的,毕竟公开密钥加密的数据,只有对应的私钥才可以解密,而这个私钥一开始就在服务端手里而不在黑客手里
image

我曾经以为这样就万无一失了,文章也就到此结束了,可以和血包杀手愉快的timi了,可是,你有没有听说过,中间人攻击

中间人攻击

黑客拦截”用公开加密密钥机密后的共享密钥“后不是解密不了吗,好,那我就不拦截这个了,我拦截第一个请求好吧,我拦截服务端传给你的公开密钥,我拦截到了,我再给你个假的,(像极了《让子弹飞》中,张麻子与马邦德的关系,出任鹅城县长)。从根上就伪装成你,以后就等于我是个中间人(中转站),所有的请求,数据都要经过我,那我就可以记录下来其中你的敏感数据,可怕。

image
  • 其实中间人攻击还要有好多种,以后有机会写一写,我们先大概了解下是什么意思就好~

数字证书

  • 所以现在问题又到了这里,我无法确保这个公钥是服务端发给我的,还是中间人发给我的?可这世上没有用钱解决不了的问题,虽然我确保不了,可是有人可以确保,就是得花钱,我们可以使用由数字证书认证机构(CA)和其他相关机构颁发的公开密钥证书
  • 数字证书认证机构处于客户端和服务器双方都信赖的第三方机构的立场上。有兴趣的同学可以自行去了解下~
  • 所以HTTPS靠非对称性加密及数字证书保证了安全性

写在最后

总结

  • 此文章从HTTP报文结构开始,到HTTP首部,到返回状态码,到cookie,再延伸到HTTPS加密方式,每一部分都进行了详细的介绍,希望对大家有用!

絮絮叨叨

  • 如果大家觉得这篇文章对自己有一点点帮助的话,欢迎关注此号 java小杰要加油
  • 原创不易,实需鼓励

若文章有误欢迎指出,靓仔靓女们,我们下篇文章见,关注我,开启我们的故事

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容