HTTP 超文本传输协议(Hyper Text Transfer Protocol)是一个应用层的协议,使用相当广泛,比如我们常说浏览器敲入网址打开网页,浏览器跟后台服务器之间就用的是HTTP,并且也常用于后端各个微服务之间的数据请求和通信。是我们常说的Web技术的基石,也是互联网的基础技术之一。
HTTP的一些优点:
HTTP是W3C和IETF两个国际组织共同推出的标准化计算协议,目前主流的是1.1版本,2.0版本也在逐步推广中,想要查阅协议原文的同学可以看这里:
HTTP1.1协议(RFC 2616): https://datatracker.ietf.org/doc/html/rfc2616
HTTP2.0协议(RFC 7540): https://datatracker.ietf.org/doc/html/rfc7540
2.0在网络传输技术,报文细节方面有很多优化。但对于开发同学来讲,关联度并不大,所以后续将以目前主流的1.1版协议内容进行介绍。
对于每一次的网络通信,一定会有发出的数据和返回,在HTTP中就叫做请求(Request)和响应(Response)。比如通过浏览器访问网页,我们要查看的网页url、请求参数等,都在Request对象中,返回的网页内容等,在Response对象中。
客户端发送一个HTTP请求到服务器的请求消息包括以下格式:请求行(request line)、请求头部(header)、空行和请求数据四个部分组成,下图给出了请求报文的一般格式。
包含 请求方法(Method) 请求地址(Request-URI ) 协议版本(HTTP-Version ) 三部分信息。举例:
GET https://www.taobao.com HTTP/1.1
就是我们常说的网址,学名是统一资源定位系统(uniform resource locator;URL)。它一般包含以下几部分:
<请求协议>://<用户名>:<密码>@<域名>:<端口>/
header一般包含的是针对这次请求的附加信息,他不是请求的实际数据。是针对数据的说明,有时也会携带cookie或者token等用于鉴权的信息。
举例:
是实际的数据部分。他的数据格式一般在header中的content-type中进行说明,常见以下几种:
HTTP响应也由四个部分组成,分别是:状态行、消息报头(Response Header)、空行和响应正文。
状态行介绍
HTTP 状态码由三个十进制数字组成,第一个十进制数字定义了状态码的类型。响应分为五类:信息响应(100–199),成功响应(200–299),重定向(300–399),客户端错误(400–499)和服务器错误 (500–599):
分类 | 分类描述 |
1** | 信息,服务器收到请求,需要请求者继续执行操作 |
2** | 成功,操作被成功接收并处理 |
3** | 重定向,需要进一步的操作以完成请求 |
4** | 客户端错误,请求包含语法错误或无法完成请求 |
5** | 服务器错误,服务器在处理请求的过程中发生了错误 |
常见状态码说明可以参考:https://www.runoob.com/http/http-status-codes.html
对返回内容的说明性描述,一些常见的Response Header举例
应答头 | 说明 |
Allow | 服务器支持哪些请求方法(如GET、POST等)。 |
Content-Encoding | 文档的编码(Encode)方法。只有在解码之后才可以得到Content-Type头指定的内容类型。利用gzip压缩文档能够显著地减少HTML文档的下载时间。Java的GZIPOutputStream可以很方便地进行gzip压缩,但只有Unix上的Netscape和Windows上的IE 4、IE 5才支持它。因此,Servlet应该通过查看Accept-Encoding头(即request.getHeader("Accept-Encoding"))检查浏览器是否支持gzip,为支持gzip的浏览器返回经gzip压缩的HTML页面,为其他浏览器返回普通页面。 |
Content-Length | 表示内容长度。只有当浏览器使用持久HTTP连接时才需要这个数据。如果你想要利用持久连接的优势,可以把输出文档写入 ByteArrayOutputStream,完成后查看其大小,然后把该值放入Content-Length头,最后通过byteArrayStream.writeTo(response.getOutputStream()发送内容。 |
Content-Type | 表示后面的文档属于什么MIME类型。Servlet默认为text/plain,但通常需要显式地指定为text/html。由于经常要设置Content-Type,因此HttpServletResponse提供了一个专用的方法setContentType。 |
Date | 当前的GMT时间。你可以用setDateHeader来设置这个头以避免转换时间格式的麻烦。 |
Expires | 应该在什么时候认为文档已经过期,从而不再缓存它? |
Last-Modified | 文档的最后改动时间。客户可以通过If-Modified-Since请求头提供一个日期,该请求将被视为一个条件GET,只有改动时间迟于指定时间的文档才会返回,否则返回一个304(Not Modified)状态。Last-Modified也可用setDateHeader方法来设置。 |
Location | 表示客户应当到哪里去提取文档。Location通常不是直接设置的,而是通过HttpServletResponse的sendRedirect方法,该方法同时设置状态代码为302。 |
Refresh | 表示浏览器应该在多少时间之后刷新文档,以秒计。除了刷新当前文档之外,你还可以通过setHeader("Refresh", "5; URL=http://host/path")让浏览器读取指定的页面。 注意这种功能通常是通过设置HTML页面HEAD区的<META HTTP-EQUIV="Refresh" CONTENT="5;URL=http://host/path">实现,这是因为,自动刷新或重定向对于那些不能使用CGI或Servlet的HTML编写者十分重要。但是,对于Servlet来说,直接设置Refresh头更加方便。 注意Refresh的意义是"N秒之后刷新本页面或访问指定页面",而不是"每隔N秒刷新本页面或访问指定页面"。因此,连续刷新要求每次都发送一个Refresh头,而发送204状态代码则可以阻止浏览器继续刷新,不管是使用Refresh头还是<META HTTP-EQUIV="Refresh" ...>。 注意Refresh头不属于HTTP 1.1正式规范的一部分,而是一个扩展,但Netscape和IE都支持它。 |
Server | 服务器名字。Servlet一般不设置这个值,而是由Web服务器自己设置。 |
Set-Cookie | 设置和页面关联的Cookie。Servlet不应使用response.setHeader("Set-Cookie", ...),而是应使用HttpServletResponse提供的专用方法addCookie。参见下文有关Cookie设置的讨论。 |
这里面就是实际的请求数据了。header中的content-type了,决定了返回数据的类型。浏览器也会基于这个,选择如何处理返回数据。content-type参考:https://www.runoob.com/http/http-content-type.html
在软件开发中,网络通信一般会选择应用层的HTTP,或者基于TCP的HSF、DUBBO等协议的方式进行。
在公网传输中一般使用HTTP,而在后端服务器之间,有时会使用HSF、Dubbo等协议进行网络通信。
这里对他们进行一下对比
HTTP | HSF、Dubbo等 | |
网络层级 | 应用层 | 传输层 |
数据结构 | 协议中规定了请求和相应的格式 | 每种协议各不相同 |
通信效率 | 作为标准协议冗余信息较多,通信效率相对较低 | 通过自定义数据结构压缩数据内容,通信效率较高 |
是否面向互联网 | 是,Web中通行的标准协议 | 否,面向特定场景使用 |
安全性 | 通过SSL加强 | 一般用于内网,也可以通过SSL等技术进行通信加密 |
适用场景 | 一般性的Web开发和互联网应用 | 对性能有要求的特定场景 |
Spring MVC是目前常用的基于JAVA语言的Web开发框架。在controller中,我们会声明对哪些HTTP请求进行处理,并且返回的数据是什么样的,这时就会用到一些注解,这里配合前面介绍过的HTTP协议内容进行说明。
常用注解举例:
常用注解举例:
在实际的项目开发中,后端同学经常需要向前端同学提供接口说明文档。既然我们上面已经提到HTTP是一种标准的同学协议,内容也不复杂,并且相当多的部分已经通过Spring MVC注解体现出来了。是否能够有一种方式自动生成接口说明文档呢?答案是肯定的,我们一般通过Swagger实现这个功能,Swagger发展到第三个版本时,更名为OpenAPI 3.0,所以OpenApi和Swagger目前可以认为是同一个东西。
Swagger的介绍可以看这里:https://swagger.io 。他实际上有接口设计工具、代码生成工具、文档UI查看器等多个部分组成,这里以开发中常用的Swagger UI进行介绍
使用时需要先引入jar包,然后进行参数配置,最后,在程序启动时,就能够在程序运行时,看到swagger格式的接口文档了。由于篇幅原因,这里不详细介绍配置过程,看下最终的结果举例:
如果只是用基于SpringMVC注解生成的文档,会发现缺少一些接口和字段细节信息,这里介绍一些常见的Swagger注解用于给生成的文档中补充信息。
@Api:修饰整个类,描述Controller 的作用
@ApiOperation:描述一个类的一个方法,或者说一个接口
@ApiParam:单个参数描述
@ApiModel:用对象来接收参数
@ApiProperty:用对象接收参数时,描述对象的一个字段
看到这里HTTP相关的内容介绍就差不多了,大家可能会有疑问,作为软件开发工程师,我又不是搞通信的,HTTP协议对我有什么用呢?这里我基于自己的经验给大家一些建议:
基于上面推荐的前后端以及测试的并行工作模式,我所在的大禹平台团队,做了有针对性的功能开发,以我们的功能举例:
可以通过页面录入或者导入的方式得到HTTP的接口定义。针对不同的场景,可以有多种使用方式:
感谢您能看到这里,我想您对HTTP已经有了较为全面的了解,并且对HTTP在实际开发工作中的作用有了更深的理解。HTTP作为比较基础的Web通用技术,有着非常广泛的使用,请愉快地使用起来吧
https://www.runoob.com/http/http-messages.html
https://www.runoob.com/http/http-header-fields.html
https://www.runoob.com/http/http-status-codes.html
本文为阿里云原创内容,未经允许不得转载。