关于HTTP请求你需要知道的一切

简介: 前言现在几乎所有的客户端(不管是APP也好,H5页面也好),都需要用到网络请求,而绝大多数公司都会把客户端的网络框架封装好,我们直接拿来用就好了,不用管什么原理,只要请求以后能得到正确的返回就可以了,但如果让你从零开始封装一个请求框架,或者你接到了一个别的项目,当前的网络请求框架不合适或者不好用了,这时候你可能就懵了。

前言

现在几乎所有的客户端(不管是APP也好,H5页面也好),都需要用到网络请求,而绝大多数公司都会把客户端的网络框架封装好,我们直接拿来用就好了,不用管什么原理,只要请求以后能得到正确的返回就可以了,但如果让你从零开始封装一个请求框架,或者你接到了一个别的项目,当前的网络请求框架不合适或者不好用了,这时候你可能就懵了。本文将会讲述请求(Request)和响应(Response)的完整组成部分、Header的关键参数、HTTP 状态码的含义以及如何封装自己的请求框架。

HTTP Request请求报文

先上个图

img_54c458ebeaedf9c1692a9acb6703ba43.png
请求报文 Request

分为三个部分:请求行、Headers和Body。
请求行:又分为三个部分method(下面会详细讲)、path(给服务器看得,如果你会一些后台的知识你会明白这其实就是来找后台的响应方法的)和HTTP version(当前大多数都是1.1,当然2.0会是趋势)
Headers:headers上的信息可以有很多,不止是图中列的这些,比较重点是Content-Type(下面会详细讲,现在知道是指body的类型就可以),还有很多自己公司后台定义的参数比如token、stamptime等等。
Body:首先body不是一定存在的,这需要根据你请求行里的方法决定,如果是POST或者PUT那么body一定存在;如果是GET或者DELETE那么则没有body,body的类型刚才也提了由headers里面的Content-Type决定。
注意:
以上就是一个完整的Request的组成部分,这里还需要再提一下URL,一个完整的URL组成部分: 协议类型服务器地址路径。那么根据上图的信息,请求的完整URL就是: http://api.github.com/users
下面我用Retrofit为例,模拟一下上面的请求

 Retrofit retrofit = new Retrofit.Builder()
                .baseUrl("http://api.github.com")
                .build();

    @Headers("Content-Type:text/plain")
    @POST("/users")
    Call<User> getUser(@Field("username")String name,@Field("pwd")String pwd);

retrofit.create(KakaService.class).getUser("kaka","123456");

这一段代码发给服务端过后就会变成上面HTTP Request请求报文,这下关于Request请求的组成应该都明白了吧

HTTP Response响应报文

再上个图


img_f06c1561e68a8b27032f08ee71e71261.png
Response响应报文

同样分为三个部分:状态行、Headers和Body
状态行:三个部分,HTTP version、status code和status message。status 是状态码,对请求结果做描述的,下面会详细讲。
Headers:响应返回的头信息,其中比较常见的是Content-Type、Content-Length和Cache-Control等等,响应返回的headers包含的参数很多,这里只举了一小部分,想看更多的可以使用POSTMAN请求一下。
Body:这是我们开发者最最关心的内容,来自服务器的相应信息。返回内容现在一般都是json了(不过还是有一些公司返回的xml),通过json相关的解析工具转成我们想要的数据。

Request 请求Method

GET

⽤于获取资源
对服务器数据不进⾏修改
不发送 Body

HTTP请求报文
GET /users/1 HTTP/1.1
Host: api.github.com
//代码展示
@GET("/users/{id}")
Call<User> getUser(@Path("id") String id);

POST

⽤于增加或修改资源
发送给服务器的内容写在 Body ⾥⾯

HTTP 请求报文
POST /users HTTP/1.1
Host: api.github.com
Content-Type: application/x-www-form-urlencoded
Content-Length: 13
username=kaka&pwd=123456
代码展示
@FormUrlEncoded
@POST("/users")
Call<User> addUser(@Field("username") String username, @Field("pwd") String
pwd)

PUT

⽤于修改资源
发送给服务器的内容写在 Body ⾥⾯

HTTP 请求报文
PUT /users/1 HTTP/1.1
Host: api.github.com
Content-Type: application/x-www-form-urlencoded
Content-Length: 13
pwd=654321
代码展示
@FormUrlEncoded
@PUT("/users/{id}")
Call<User> updateGender(@Path("id") String id, @Field("pwd") String
pwd);

DELETE

⽤于删除资源
不发送 Body

HTTP 请求报文
DELETE /users/1 HTTP/1.1
Host: api.github.com
@DELETE("/users/{id}")
Call<User> getUser(@Path("id") String id);

HEAD

和 GET 使⽤⽅法完全相同
和 GET 唯⼀区别在于,返回的响应中没有 Body
一般用在多线程下载的时候,header里面返回Range / Accept-Range里的数据可以判断下载范围
注意:
本文只讲一些常用的重点方法,像HEAD、PATCH等方法不讲可以自行百度。

Header ⾸部

Content-Type

指定 Body 的类型。主要有四类:

  1. text/html
    请求 Web ⻚⾯是返回响应的类型, Body 中返回 html ⽂本。格式如下:
HTTP/1.1 200 OK
Content-Type: text/html; charset=utf-8
Content-Length: 853
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
......
  1. x-www-form-urlencoded
    Web ⻚⾯纯⽂本表单的提交⽅式。


    img_f53e0eac6552baf46121fcc014f2bde6.png
    纯文本提交方式
POST /users HTTP/1.1
Host: api.github.com
Content-Type: application/x-www-form-urlencoded
Content-Length: 27
name=kaka&pwd=123456
  1. multitype/form-data

    Web ⻚⾯含有⼆进制⽂件时的提交⽅式。
    img_02dac2882ab919cc83c8de51ad8834a6.png
    含有二进制文件时的提交
POST /users HTTP/1.1
Host: hencoder.com
Content-Type: multipart/form-data; boundary=----
WebKitFormBoundary7MA4YWxkTrZu0gW
Content-Length: 2382
------WebKitFormBoundary7MA4YWxkTrZu0gW
Content-Disposition: form-data; name="name"
rengwuxian
------WebKitFormBoundary7MA4YWxkTrZu0gW
Content-Disposition: form-data; name="avatar"; filename="avatar.jpg"
Content-Type: image/jpeg
JFIFHHvOwX9jximQrWa......对应 Retrofit 的代码:
4. application/json , image/jpeg , application/zip ...
单项内容(⽂本或⾮⽂本都可以),⽤于 Web Api 的响应或者 POST / PUT 的请求
请求中提交 JSON
对应 Retrofit 的代码:
响应中返回 JSON
------WebKitFormBoundary7MA4YWxkTrZu0gW
  1. application/json , image/jpeg , application/zip ...
    单项内容(⽂本或⾮⽂本都可以),⽤于 Web Api 的响应或者 POST / PUT 的请求

注意:header里面的这个content-type还是非常重要的,这决定于你和后台商量好的请求格式,以上的这些请牢记!

Content-Length

指定 Body 的⻓度(字节)

Location

指定重定向的⽬标 URL

User-Agent

⽤户代理,即是谁实际发送请求、接受响应的,例如⼿机浏览器、某款⼿机 App。

Range / Accept-Range

按范围取数据
Accept-Range: bytes 响应报⽂中出现,表示服务器⽀持按字节来取范围数据
Range: bytes=<start>-<end> 请求报⽂中出现,表示要取哪段数据
Content-Range:<start>-<end>/total 响应报⽂中出现,表示发送的是哪段数据
作⽤:断点续传、多线程下载。

其他 Headers

Accept: 客户端能接受的数据类型。如 text/html
Accept-Charset: 客户端接受的字符集。如 utf-8
Accept-Encoding: 客户端接受的压缩编码类型。如 gzip
Content-Encoding:压缩类型。如 gzip

Status Code 状态码

三位数字,⽤于对响应结果做出类型化描述(如「获取成功」「内容未找到」)。
1xx:临时性消息。如: 100 (继续发送)、 101(正在切换协议)
2xx:成功。最典型的是 200( OK)、 201(创建成功)。
3xx:重定向。如 301(永久移动)、 302(暂时移动)、 304(内容未改变)。
4xx:客户端错误。如 400(客户端请求错误)、 401(认证失败)、 403(被禁⽌)、 404(找不到内容)。
5xx:服务器错误。如 500(服务器内部错误)
注意:为什么会有status code?status code的出现是为了开发者调试的,其中200我们最喜欢也最常见,返回正常,一切OK。其中最主要记得分清楚4XX和5XX的区别,但凡是4XX的这种移动端的同事就不要找后台的麻烦,指定是客户端的问题了,1XX临时性消息一般是迅雷多线程下载会返回的,3XX是重定向,一般用的比较少。

如何封装自己的网络框架

其实,我相信你只要认真学习了上面的知识,封装自己的网络请求框架应该不是什么问题了,下面我们就来总结一下,都需要注意封装是什么:
1.headers:这个是必须要注意的,因为现在的网络请求越来越注意安全性,有一些中小型的创业公司为了信息安全,经常在headers传一系列的加密信息,因为对headers的封装是必不可少的
2.body:body里面的内容封装主要取决于headers里的Content-type,根据不同的Content-type生成不同的body可以让我们的网络框架使用更加方便。
3.status code:对status code的封装是方便当我们网络发生错误时,给我们开发者更快地定位问题所在。
最后给大家提供一个我自己封装的例子,用的是当前比较流行的Retrofit+RxJava,https://www.jianshu.com/p/b5546905ccbc

目录
相关文章
|
18天前
|
数据采集
Haskell爬虫:连接管理与HTTP请求性能
Haskell爬虫:连接管理与HTTP请求性能
|
25天前
|
JSON 安全 前端开发
类型安全的 Go HTTP 请求
类型安全的 Go HTTP 请求
|
1天前
|
监控 网络协议 应用服务中间件
【Tomcat源码分析】从零开始理解 HTTP 请求处理 (第一篇)
本文详细解析了Tomcat架构中复杂的`Connector`组件。作为客户端与服务器间沟通的桥梁,`Connector`负责接收请求、封装为`Request`和`Response`对象,并传递给`Container`处理。文章通过四个关键问题逐步剖析了`Connector`的工作原理,并深入探讨了其构造方法、`init()`与`start()`方法。通过分析`ProtocolHandler`、`Endpoint`等核心组件,揭示了`Connector`初始化及启动的全过程。本文适合希望深入了解Tomcat内部机制的读者。欢迎关注并点赞,持续更新中。如有问题,可搜索【码上遇见你】交流。
【Tomcat源码分析】从零开始理解 HTTP 请求处理 (第一篇)
|
24天前
|
数据采集 JSON API
异步方法与HTTP请求:.NET中提高响应速度的实用技巧
本文探讨了在.NET环境下,如何通过异步方法和HTTP请求提高Web爬虫的响应速度和数据抓取效率。介绍了使用HttpClient结合async和await关键字实现异步HTTP请求,避免阻塞主线程,并通过设置代理IP、user-agent和cookie来优化爬虫性能。提供了代码示例,演示了如何集成这些技术以绕过目标网站的反爬机制,实现高效的数据抓取。最后,通过实例展示了如何应用这些技术获取API的JSON数据,强调了这些方法在提升爬虫性能和可靠性方面的重要性。
异步方法与HTTP请求:.NET中提高响应速度的实用技巧
|
10天前
|
JSON JavaScript 前端开发
Haskell中的数据交换:通过http-conduit发送JSON请求
Haskell中的数据交换:通过http-conduit发送JSON请求
|
12天前
|
JSON API 开发者
Python网络编程新纪元:urllib与requests库,让你的HTTP请求无所不能
【9月更文挑战第9天】随着互联网的发展,网络编程成为现代软件开发的关键部分。Python凭借简洁、易读及强大的特性,在该领域展现出独特魅力。本文介绍了Python标准库中的`urllib`和第三方库`requests`在处理HTTP请求方面的优势。`urllib`虽API底层但功能全面,适用于深入控制HTTP请求;而`requests`则以简洁的API和人性化设计著称,使HTTP请求变得简单高效。两者互补共存,共同推动Python网络编程进入全新纪元,无论初学者还是资深开发者都能从中受益。
31 7
|
10天前
|
开发者
HTTP状态码是由网页服务器返回的三位数字响应代码,用于表示请求的处理结果和状态
HTTP状态码是由网页服务器返回的三位数字响应代码,用于表示请求的处理结果和状态
17 1
|
21天前
|
缓存 网络协议 安全
揭秘浏览器背后的神秘之旅:一网打尽HTTP请求流程,让你网络冲浪更顺畅!
【8月更文挑战第31天】当在浏览器中输入网址并按下回车键时,一系列复杂的HTTP请求流程随即启动。此流程始于DNS解析,将域名转化为IP地址;接着是与服务器的TCP三次握手建立连接。连接建立后,浏览器发送HTTP请求,其中包含请求方法、资源及版本等信息。服务器接收请求并处理后返回HTTP响应,包括状态码、描述及页面内容。浏览器解析响应,若状态码为200则渲染页面,否则显示错误页。整个流程还包括缓存处理和HTTPS加密等步骤,以提升效率和保障安全。理解该流程有助于更高效地利用网络资源。通过抓包工具如Wireshark,我们能更直观地观察和学习这一过程。
34 4
|
20天前
|
JSON 监控 API
http 请求系列
XMLHttpRequest(XHR)是一种用于在客户端和服务器之间进行异步HTTP请求的API,广泛应用于动态更新网页内容,无需重新加载整个页面。本文提供了多个官方学习资源,包括MDN Web Docs、WhatWG和W3C的规范文档,涵盖属性、方法、事件及示例代码。XHR的主要应用场景包括动态内容更新、异步表单提交、局部数据刷新等,具有广泛的支持和灵活性,但也存在处理异步请求的复杂性等问题。最佳实践包括使用异步请求、处理请求状态变化、设置请求头、处理错误和超时等。这些资源和实践将帮助你更好地理解和使用XHR。
20 1
|
1月前
|
JSON API 数据格式
Python网络编程:HTTP请求(requests模块)
在现代编程中,HTTP请求几乎无处不在。无论是数据抓取、API调用还是与远程服务器进行交互,HTTP请求都是不可或缺的一部分。在Python中,requests模块被广泛认为是发送HTTP请求的最简便和强大的工具之一。本文将详细介绍requests模块的功能,并通过一个综合示例展示其应用。

热门文章

最新文章