美文网首页计算机网络面试管理干货
你应该知道的HTTP基础知识

你应该知道的HTTP基础知识

作者: 怪盗kidou | 来源:发表于2016-05-04 19:10 被阅读35106次

    转载请注明出处
    作者:@怪盗kidou
    说明:本文内容仅限于本人熟知的内容,HTTP的内容任意一个知识点都可以单独写一篇博客,所以别指望靠一篇博客可以讲清楚,本文的主要作用是为以后的博客作铺垫,所以更详细的HTTP协议内容可参考RFC 2616,本人水平有限,如有不正确的地方欢迎留言指出。

    本文主要内容:

    • HTTP请求报文格式
    • HTTP响应报文格式
    • Header
    • 请求体的3种形式
    • 推荐调试工具
    • HTTP的组成图示

    1. HTTP请求报文格式

    HTTP 的请求报文分为三个部分 请求行请求头请求体,格式如图:

    HTTP请求报文格式

    注:部分文章也将HTTP请求报文分为两部分请求头和请求体,请求头的第一行为请求行。

    1.1 请求行

    请求行(Request Line)分为三个部分:请求方法请求地址协议及版本,以CRLF(\r\n)结束。
    HTTP/1.1 定义的请求方法有8种:GET、POST、PUT、DELETE、PATCH、HEAD、OPTIONS、TRACE,最常的两种GET和POST,如果是RESTful接口的话一般会用到GET、POST、DELETE、PUT。

    在了解请求地址之前,先了解一下URL的构成:


    URL.png

    PATH是URL主机以后的部分,即包含了Query String,如:

    URL PATH
    http://http://www.jianshu.com/ /
    http://localhost:8080/index.php?id=1234 /index.php?id=1234
    http://weibo.com/920507888/home /920507888/home

    请求报文示例:


    HTTP请求报文格式示例

    2. HTTP响应报文格式

    HTTP响应的格式上除状态行(第一行)与请求的请求行不一样以外,其它就格式而言是一样的,但排除状态行和请求行的区别,从Header上还是能区分出HTTP请求和HTTP响应的,怎么区分就要看前面的常见Header啦。

    HTTP_响应

    状态码(就是上图中的响应码) 的详细可以查看 HTTP状态码详解

    为了更直观的展示HTTP响应,下面的例子,自nc 1270.0.1:80 << EOFEOF之间,是一个简单的HTTP请求.

    HTTP请求和响应

    3. Header

    Header可用于传递一些附加信息,格式: 键: 值,注意冒号后面有一个空格!如:

    Content-Length: 1024
    Content-Type: text/plain
    

    3.1 请求和响应常见通用Header

    名称 作用
    Content-Type 请求体/响应体的类型,如:text/plain、application/json
    Accept 说明接收的类型,可以多个值,用,(半角逗号)分开
    Content-Length 请求体/响应体的长度,单位字节
    Content-Encoding 请求体/响应体的编码格式,如gzip,deflate
    Accept-Encoding 告知对方我方接受的Content-Encoding
    ETag 给当前资源的标识,和Last-ModifiedIf-None-MatchIf-Modified-Since配合,用于缓存控制
    Cache-Control 取值为一般为no-cachemax-age=XX,XX为个整数,表示该资源缓存有效期(秒)

    3.2 常见请求Header

    名称 作用
    Authorization 用于设置身份认证信息
    User-Agent 用户标识,如:OS和浏览器的类型和版本
    If-Modified-Since 值为上一次服务器返回的 Last-Modified 值,用于确认某个资源是否被更改过,没有更改过(304)就从缓存中读取
    If-None-Match 值为上一次服务器返回的 ETag 值,一般会和If-Modified-Since一起出现
    Cookie 已有的Cookie
    Referer 表示请求引用自哪个地址,比如你从页面A跳转到页面B时,值为页面A的地址
    Host 请求的主机和端口号

    3.3 常见响应Header

    名称 作用
    Date 服务器的日期
    Last-Modified 该资源最后被修改时间
    Transfer-Encoding 取值为一般为chunked,出现在Content-Length不能确定的情况下,表示服务器不知道响应版体的数据大小,一般同时还会出现Content-Encoding响应头
    Set-Cookie 设置Cookie
    Location 重定向到另一个URL,如输入浏览器就输入baidu.com回车,会自动跳到 https://www.baidu.com ,就是通过这个响应头控制的
    Server 后台服务器

    4. 请求体的3种形式

    根据应用场景的不同,HTTP请求的请求体有三种不同的形式。

    第一种:

    移动开发者常见的,请求体是任意类型,服务器不会解析请求体,请求体的处理需要自己解析,如 POST JSON时候就是这类。


    HTTP请求,形式一

    第二种:

    第二种和第三种都有固定格式的,是服务器端开发人员最先了解到的两种。这里的格式要求就是URL中Query String的格式要求:多个键值对之间用&连接,键与值之前用=连接,且只能用ASCII字符,非ASCII字符需使用UrlEncode编码。

    HTTP请求,形式二 第二种示例

    第三种:

    第三种请求体的请求体被分成为多个部分,文件上传时会被使用,这种格式最先应该是被用于邮件传输中,每个字段/文件都被boundary(Content-Type中指定)分成单独的段,每段以-- 加 boundary开头,然后是该段的描述头,描述头之后空一行接内容,请求结束的标制为boundary后面加--,结构见下图:

    HTTP请求,形式三

    区分是否被当成文件的关键是Content-Disposition是否包含filename,因为文件有不同的类型,所以还要使用Content-Type指示文件的类型,如果不知道是什么类型取值可以为application/octet-stream表示该文件是个二进制文件,如果不是文件则Content-Type可以省略。

    下图为一个带有文件的上传的请求体原文:


    第三种请求体示例

    注意:
    第二、三种请求体需求配合特定的Content-Type请求头,如:
    第二种配合Content-Type:application/x-www-form-urlencoded
    第三种配合Content-Type: multipart/form-data; boundary={boundary}
    *上面的form-data也可以是mixedalternativedigestparallel,但我只用到了form-data
    如果两者没有相配合,那么服务器不会解析请求体,也就是说只会当成第一种情况!

    表单或者模拟表单 指的就是第二种和第三种(multipart/form-data)

    5. 推荐调试工具

    5.1 cURL

    cURL 相当强大命令行工具,基本上你知道的上层协议它都支持,具体使用方法就自行发现了,下图为发起一个HTTP GET请求并打印请求和响应的详细内容。

    cURL截图

    5.2 bat

    batastaxie(谢孟军) 用Golang开发的类似cURL的命令行API调试工具,可以方便的打印出HTTP请求和响应,还能高亮Header、格式化JSON等功能,非常好用,API调式神器。

    bat截图

    5.3 nc

    nc 是 netcat 的简写,被称为“网络工具中的瑞士军刀”,不过我个人是把它当成Socket用,经常使用它来打印各种请求,当然它的作用可不只是这样,你也可以用它发起各式各样的请求,以前调式POP3也是用的它,只不过请求报文得自己写,上面有个图“HTTP请求和响应”就是用nc完成的。

    6. HTTP的组成图示

    感觉写得有点乱,如果按下图的结构写会不会更好呢?

    HTTP报文组成.png

    相关文章

      网友评论

      • 爱在记忆消失前:不懒,不赖!
      • ffaf8cc1076e:虽然有点看不懂,但写的挺好的,自己的努力。
        小默森:不明觉厉
      • 德鲁大叔凯里欧文:对于android程序员来说这篇文章介绍HTTP相关足够用了
      • 学费:啊,一下子好难记住啊
      • nic启立:了解足够了!谢谢。
      • CH_DHY:总结得非常棒
      • 月下大象:写的不错
      • 键盘上的麒麟臂:请求文件时候的响应体是怎么样的
        怪盗kidou:@键盘上的麒麟臂 就是文件的内容嘛,假如文件里面的内容是 123 的话,那响应体里就是写的 123
        键盘上的麒麟臂:@怪盗kidou 不太了解:dizzy_face:
        怪盗kidou:响应体是文件的二进制数据(如果没有使用传输编码或压缩的话)
      • luckbing:大神,看不懂的应该如何恶补,需要看网络基础知识么?就是对里面的专业词汇不懂,没有一个把我(就像生活中说杯子,我脑海中就会浮现出杯子的样子),但是你这里说的比如请求头,我就不知道是啥,用来干嘛的,具体有什么用,请教下这些在哪里有得系统的补下,推荐书籍最好了,先谢谢大神:grin:
        luckbing:@怪盗kidou 我是电子专业的!所以计算机基础很薄弱!!!现在在努力ing...谢谢哈:relaxed:
        怪盗kidou:你不是计算机专业出身么,如果是的话,这些在大学的《TCP/IP》教材里就有,其它的话我就没有看过其它什么专业的书,都是直接Google相关的内容就行,如果只是HTTP的内容,可以看看 《图解HTTP》,可能更容易入门一点
      • 十旋转45度:谢谢博主分享~如果能把最后一张图示的字变清晰点就更好啦:+1:
        怪盗kidou:@十旋转45度 被压缩了:flushed:
      • BlackBeard:这些API调试工具怎么用的,比如这个bat,我百度到的都是批处理,安装教程是怎样的,这个bat
        BlackBeard:@怪盗kidou 谢谢回复哈,学习啦
        怪盗kidou:@BlackBeard 下载下来就是exe(windows上),不用安装,把他放到 PATH 环境变量里,或者直接用cmd输入完整路径都行
      • 周娱:推荐一个博客 http://www.chinacion.cn
      • 吃货打代码:最后一个脑图就够了:flushed:
      • 壹公升:我想问下mac怎么安装bat啊?
        怪盗kidou:@壹公升 https://github.com/astaxie/bat/releases 里面直接下 _darwin_amd64.zip 后缀的就行
      • longzekai:写的很详细啊。
      • developerHaoz:不知道大神什么时候有空写一下,你以前说过的 反射 和 泛型,一直很期待
        怪盗kidou:@DeveloperHaoz 其实不难,就是写出来还挺不好写的
        developerHaoz:@怪盗kidou 那就过年好好休息一下,指不定过完年就有feel了,然后我又有干货看了:smile:
        怪盗kidou:有点写不出来,没找到感觉
      • developerHaoz:思路很清晰啊,刚好下学期上计算机网络,当是先预习一下,而且感觉对最近看Volley源码的理解也有一定的帮助,谢谢啦
        怪盗kidou:@DeveloperHaoz 打好基础吧,基础是很重要的
        developerHaoz:@怪盗kidou 现在还有很多要学的,乘着寒假给自己充充电,不知道大神有什么建议吗?本人现在大二,以后打算走Android开发
        怪盗kidou:有前途:+1:
      • 5b66957dcf83:写得很好,让我对HTTP概念终于有个大致的掌握
      • 丶萤火之森:工具不错啊,怎么用呢
      • 1cf2c90a5564:看了一遍没看懂,看到项目里header字段后面加了个空格,然后就全看懂了。没搞过网络的人伤不起 :joy:
        怪盗kidou:@Yaren_Android 其实就是个http1.x协议就是带格式的文本
      • 得病的人:先马再看
      • Neogx:ENEN··不错,通俗易懂!
      • chen_666:写的挺好,接着写啊
      • 113bdaafbb89:问一下 如何高亮Header、格式化JSON数据 呢?看了你的这篇文章后我也安装了bat,可是不知道怎么高亮Header,格式化Json数据
        怪盗kidou:@果果四分卫 我猜测这个应该是Content-Type的缘故,你的这个接口返回的Content-Type是text/html,那么bat不认为它是json,所以没有格式化 :grin:
        113bdaafbb89:哦哦,那就不方便了,高亮无所谓,但是在window上不格式化json数据就有点郁闷了,我在window上看到的请求结果是这样子的:
        $ bat http://ip.taobao.com/service/getIpInfo.php?ip=202.202.33.33
        GET /service/getIpInfo.php?ip=202.202.33.33 HTTP/1.1
        Host: ip.taobao.com
        Accept: application/json
        Accept-Encoding: gzip, deflate
        User-Agent: bat/0.1.0




        HTTP/1.1 200 OK
        Content-Type : text/html
        Connection : keep-alive
        Vary : Accept-Encoding
        X-Powered-By : PHP/5.4.31
        Content-Encoding : gzip
        Server : Tengine
        Date : Mon, 11 Jul 2016 03:08:31 GMT


        {"code":0,"data":{"country":"\u4e2d\u56fd","country_id":"CN","area":"\u897f\u5357","area_id":"500000","region":"\u91cd\u5e86\u5e02","region_id":"500000","city":"\u91cd\u5e86\u5e02","city_id":"500100","county":"","county_id":"-1","isp":"\u6559\u80b2\u7f51","isp_id":"100027","ip":"202.202.33.33"}}
        怪盗kidou:@果果四分卫 不用特别设置啥,自己就可以高亮,windows的话可能不行,cmd好像只支持两种颜色
      • c1b405d8ae82:url为什么要这边写一半,那边写一半,这样感觉好麻烦,
      • c1b405d8ae82:我用的post请求,整体的url = "http://mm.mfniu.com/themelist/essenceAsk&quot;,

        .baseUrl("http://mm.mfniu.com/&quot;);这样写,


        public interface BlogService {

        @post("themelist/essenceAsk")
        Call<ResponseBody> getFirstBlog(@Field("page") String PAGE,@field("pageSize") String String);
        }


        可运行起来总是报这个错IllegalArgumentException: @field parameters can only be used with form encoding. (parameter #1)

        大神哪里甚至的不对,假如我要是不添加参数应该怎么写呢
        8a52b918f8c8:@c1b405d8ae82 要加个@FormUrlEcoded
      • 乱尘:看到是基础,屁颠屁颠的跑进来,看完后,灰溜溜滚粗。 :joy: 还是玩C去
      • haotie1990:学到了好多 :pray:
        怪盗kidou:@haotie1990 感谢打赏 :grin: ,这些只是一小部分
      • Champion是冠军:你的文章我都蛮喜欢的,很符合我的胃口
        怪盗kidou:@Eric胡 是吗?感觉写得跟差啊
      • 无聊的ddd:很好希望再写些dagger2的
      • mingzikb:好文,支持,感谢分享。
      • Fritz_Xu:Http的知识啊,忽然想起当当4月书香节买不到图解http的郁闷
      • 748a30ef292f:写的不错
      • lynch0571:深度终端好用吗?
        怪盗kidou:@D_C 感觉还不错
      • 远行游子:可以的
      • 下一页会是幸福吗:学习了,最近学了一段时间的javaweb,一直用请求头,响应头,用http还没具体研究过,里面东西也不少啊。打算做安卓软件开发,网络还是要认真研究的啊,要不手机就是一块板砖了。
        怪盗kidou:@下一页会是幸福吗 HTTP的内容对于程序员来说还是很重要的
      • 吴悟无:我看懂了开头,然后。。。
        怪盗kidou:@吴悟无 :relaxed: ,应该不难,说白了就是固定的格式,然后加上一些约束
      • Mi氓:好文 :+1: ,学习了
      • 说dian什么好呢:虽然看的有点吃力,但还是挺喜欢的
        怪盗kidou:@说dian什么好呢 加油,程序员必备知识
      • klisly:正在学习,谢谢分享
      • 8fc333e7d319:简书里的朋友对这个熟悉的应该不多。哈哈
        be6b8e502b93:基本上做编程开发的这个都是基础知识
        怪盗kidou:@静水_流深 肯定是有高手的
      • 81afc1f44375:弱弱的问一句,这是做什么的软件呢?
        SHUTUP:@陆语Ly http:指超文本传输协议
        81afc1f44375: @怪盗kidou 嘿,表打我,我就是不知道HTTP是什么😳
        怪盗kidou:@陆语Ly 你说的哪个软件 :flushed:
      • 孙圳pm:看不懂
        怪盗kidou:@孙圳pm 说明我没讲清楚啊:joy:
      • litaotao:写得不错啊,收藏了
        iDragonfly:@怪盗kidou 确实还没讲清楚,但是实际工作对这个要求不高,所以这些了解也够用了。
        怪盗kidou:@litaotao 这还写得不错?感觉写得很乱啊

      本文标题:你应该知道的HTTP基础知识

      本文链接:https://www.haomeiwen.com/subject/mrdmlttx.html