什么是HTTP协议
HTTP是hypertext transfer protocol(超文本传输协议)的简写,它是TCP/IP协议的一个应用层协议,用于定义WEB浏览器与WEB服务器之间交换数据的过程。客户端连上web服务器后,若想获得web服务器中的某个web资源,需遵守一定的通讯格式,HTTP协议用于定义客户端与web服务器通讯的格式。
HTTP协议版本和区别
- 版本:HTTP/1.0、HTTP/1.1
- 区别:在HTTP1.0协议中,客户端与web服务器建立连接后,只能获得一个web资源。在HTTP1.1协议,允许客户端与web服务器建立连接后,在一个连接上获取多个web资源。
HTTP请求
客户端连接服务端,像服务器请求某个web资源,则称为一个http请求。
一个完整的HTTP请求包括:一个请求行,请求头,请求正文
下面是一个HTTP请求的例子:
GET/sample.jspHTTP/1.1
Accept:image/gif.image/jpeg,*/*
Accept-Language:zh-cn
Connection:Keep-Alive
Host:localhost
User-Agent:Mozila/4.0(compatible;MSIE5.01;Window NT5.0)
Accept-Encoding:gzip,deflate
username=jinqiao&password=1234
请求行
请求的第一行是“方法URL协议/版本”:GET/sample.jsp HTTP/1.1
以上代码中“GET”代表请求方法,“/sample.jsp”表示URI,“HTTP/1.1代表协议和协议的版本。
根据HTTP标准,HTTP请求可以使用多种请求方法。例如:HTTP1.1支持7种请求方法:GET、POST、HEAD、OPTIONS、PUT、DELETE和TARCE。在Internet应用中,最常用的方法是GET和POST。
URL完整地指定了要访问的网络资源,通常只要给出相对于服务器的根目录的相对目录即可,因此总是以“/”开头。
最后,协议版本声明了通信过程中使用HTTP的版本。
用户如果没有设置,默认情况下浏览器向服务器发送的都是get请求,例如在浏览器直接输地址访问,点超链接访问等都是get,用户如想把请求方式改为post,可通过更改表单的提交方式实现。
不管POST或GET,都用于向服务器请求某个WEB资源,这两种方式的区别主要表现在数据传递上:
- 如果请求方式为GET方式,则可以在请求的URL地址后以?的形式带上交给服务器的数据,多个数据之间以&进行分隔,例如:GET /mail/1.html?name=abc&password=xyz HTTP/1.1
- GET方式的特点:在URL地址后附带的参数是有限制的,其数据容量通常不能超过1K。
- 如果请求方式为POST方式,则可以在请求的实体内容中向服务器发送数据,Post方式的特点:传送的数据量无限制。
请求头
常见请求头:
- accept:浏览器通过这个头告诉服务器,它所支持的数据类型
- Accept-Charset: 浏览器通过这个头告诉服务器,它支持哪种字符集
- Accept-Encoding:浏览器通过这个头告诉服务器,支持的压缩格式
- Accept-Language:浏览器通过这个头告诉服务器,它的语言环境
- Host:浏览器通过这个头告诉服务器,想访问哪台主机
- If-Modified-Since: 浏览器通过这个头告诉服务器,缓存数据的时间
- Referer:浏览器通过这个头告诉服务器,客户机是哪个页面来的
- Connection:浏览器通过这个头告诉服务器,请求完后是断开连接还是保持连接
- User-Agent:客户端将它的操作系统,浏览器和其他属性告诉服务器
请求正文
请求头和请求正文之间是一个空行,这个行非常重要,它表示请求头已经结束,接下来的是请求正文。请求正文中可以包含客户提交的查询字符串信息:
username=jinqiao&password=1234
在以上的例子的HTTP请求中,请求的正文只有一行内容。当然,在实际应用中,HTTP请求正文可以包含更多的内容。
HTTP响应
一个http响应代表服务器端向客户端回送的数据,包括:状态行,响应头,响应正文。示例如下:
HTTP/1.1 200 OK
Server: Apache-Coyote/1.1
Content-Type: text/html;charset=ISO-8859-1
Content-Length: 105
Date: Tue, 27 May 2014 16:23:28 GMT
<html>
<head>
<title>Hello World JSP</title>
</head>
<body>
Hello World!
</body>
</html>
状态行
第一行为状态行,格式如下:HTTP版本号,状态码,原因叙述(CRLF)
状态码用于表示服务器对请求的处理结果,它是一个三位的十进制数。响应状态码分为5类,如下所示:
常见的状态码如下:
响应头
Location: 服务器通过这个头,来告诉浏览器跳到哪里
Server:服务器通过这个头,告诉浏览器服务器的型号 如:Server:Apache/1.3.6(Unix)
Content-Encoding:服务器通过这个头,告诉浏览器,数据的压缩格式
Content-Length: 服务器通过这个头,告诉浏览器回送数据的长度
Content-Language: 服务器通过这个头,告诉浏览器语言环境
Content-Type:服务器通过这个头,告诉浏览器回送数据的类型 如:Content-Type:text/html;charset=GBK
Refresh:服务器通过这个头,告诉浏览器定时刷新
Content-Disposition: 服务器通过这个头,告诉浏览器以下载方式打数据
Transfer-Encoding:服务器通过这个头,告诉浏览器数据是以分块方式回送的
Expires: -1 控制浏览器不要缓存
Cache-Control: no-cache
Pragma: no-cache
Keep-Alive:保持连接的时间 如:Keep-Alive:timeout=5,max=120
响应正文
响应正文(响应体)就是响应的消息体,如果是纯数据就是返回纯数据,如果请求的是HTML页面,那么返回的就是HTML代码,如果是JS就是JS代码,如此之类。
在服务端设置响应头来控制客户端浏览器的行为
设置Location响应头,实现请求重定向(SpringBoot实现)
@WebServlet("/ServletDemo")
public class ServletDemo extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.setStatus(302);//设置服务器响应状态
/**
* 设置响应头,服务器通过location这个头,来告诉浏览器跳转到哪里。这就是请求重定向
*/
response.setHeader("Location", "/test.html");
//response.setHeader("Location", "http://www.baidu.com"); 即可重定向到百度
/*
其实将response.setStatus和response.setHeader("Location", "/test.html");封装起来就是
response.sendRedirect();
*/
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doGet(request, response);
}
}
访问http://localhost:8080/ServletDemo,即会跳转到
其实将response.setStatus和response.setHeader("Location", "/test.html");封装起来就是 response.sendRedirect();
即可在doGet方法中使用该方法实现
response.sendRedirect("/test.html");
服务器返回一个302状态码告诉浏览器,你要的资源我没有,但是我通过Location响应头告诉你哪里有,而浏览器解析响应头Location后知道要跳转到/test.html页面,所以就会自动跳转到test.html页面。
设置Content-Encoding响应头,告诉浏览器数据的压缩格式
@WebServlet("/ServletDemo1")
public class ServletDemo1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException{
String data = "qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb"
+"qewrtyuadf"+"wertyfgnvbn"+"unjsdfbsjafsb";
System.out.println("原始数据大小为:"+data.getBytes().length);
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
GZIPOutputStream gzipOutputStream = new GZIPOutputStream(byteArrayOutputStream);
gzipOutputStream.write(data.getBytes());
gzipOutputStream.close();
//得到压缩后的数据
byte [] g = byteArrayOutputStream.toByteArray();
response.setHeader("Content-Encoding", "gzip");
response.setHeader("Content-Length", g.length+"");
response.getOutputStream().write(g);
}
@Override
protected void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{
this.doGet(request, response);
}
}
打开浏览器 访问http://localhost:8080/ServletDemo1,查看响应头
可以看到压缩格式是gzip,压缩后的大小是58
设置refresh响应头,让浏览器定时刷新
@WebServlet("/ServletDemo3")
public class ServletDemo3 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
/**
* 设置refresh响应头,让浏览器每隔3秒定时刷新
*/
// response.setHeader("refresh", "3");
/**
* 设置refresh响应头,让浏览器3秒后跳转到http://www.baidu.com
*/
response.setHeader("refresh", "3;url='http://www.baidu.com'");
response.getWriter().write("gacl");
}
@Override
protected void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{
this.doGet(request, response);
}
}
参考博文:https://www.cnblogs.com/xdp-gacl/p/3751277.html
才疏学浅之处,切莫在意。
——2019.7.10下午