细说:Http协议 篇
一、什么是http协议?
现在浏览器有许多种,比如火狐,谷歌,如果不同浏览器发送不同格式的请求,服务器对于请求解析时候,会遇到很大的麻烦。同理,如果不同的服务器返回的请求格式不一样,浏览器就要对不同的服务器做出不同的响应,压力是很大的。为了大家的方便,就出现了http协议:对浏览器客户端与服务器的数据传输格式规范化。也就是不同的浏览器发送的请求数据必须一致,不同服务器发送的响应数据必须一致。
二、查看http协议的工具
使用火狐的firebug插件(右键->firebug->网络)
三、Http请求
GET /day09/hello HTTP/1.1 -请求行 Host: localhost:8080 --请求头(多个key-value对象) User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; rv:35.0) Gecko/20100101 Firefox/35.0 Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8 Accept-Language: zh-cn,en-us;q=0.8,zh;q=0.5,en;q=0.3 Accept-Encoding: gzip, deflate Connection: keep-alive --一个空行 name=eric&password=123456 --(可选)实体内容
总共可以分为三个部分:
- 请求行
- 请求头
- 实体内容(可选,主要是post方式传送才会有的)
3.1请求行:
GET /day09/hello HTTP/1.1
从右往左的顺序进行介绍:
1、http协议版本
http1.0:当前浏览器客户端与服务器端建立连接之后,只能发送一次请求,一次请求之后连接关闭。
http1.1:当前浏览器客户端与服务器端建立连接之后,可以在一次连接中发送多次请求。
目前基本都是用http1.1
2、请求资源
首先需要明白URL、URI的区别:
URL: 统一资源定位符。http://localhost:8080/day09/testImg.html。只能定位互联网资源。是URI 的子集。
URI:统一资源标记符。/day09/hello。用于标记任何资源。可以是本地文件系统,局域网的资源(//192.168.14.10/myweb/index.html), 可以是互联网。
3、请求方式
常用的请求方式为POST、GET(在表单提交中经常使用)
GET方式提交:
地址栏(URI)会跟上参数数据。以?开头,多个参数之间以&分割。
- GET提交参数数据有限制,不超过1KB。
- GET方式不适合提交敏感密码。
- 注意: 浏览器直接访问的请求,默认提交方式是GET方式
- POST方式提交
- 参数不会跟着URI后面。参数而是跟在请求的实体内容中。没有?开头,多个参数之间以&分割。
POST提交方式
- 参数不会跟着URI后面。参数而是跟在请求的实体内容中。没有?开头,多个参数之间以&分割
- POST提交的参数数据没有限制。
- POST方式提交敏感数据。
3.2请求头
Accept: text/html,image/* -- 浏览器接受的数据类型 Accept-Charset: ISO-8859-1 -- 浏览器接受的编码格式 Accept-Encoding: gzip,compress --浏览器接受的数据压缩格式 Accept-Language: en-us,zh- --浏览器接受的语言 Host: www.it315.org:80 --(必须的)当前请求访问的目标地址(主机:端口) If-Modified-Since: Tue, 11 Jul 2000 18:23:51 GMT --浏览器最后的缓存时间 Referer: http://www.it315.org/index.jsp -- 当前请求来自于哪里 User-Agent: Mozilla/4.0 (compatible; MSIE 5.5; Windows NT 5.0) --浏览器类型 Cookie:name=eric -- 浏览器保存的cookie信息 Connection: close/Keep-Alive -- 浏览器跟服务器连接状态。close: 连接关闭 keep-alive:保存连接。 Date: Tue, 11 Jul 2000 18:23:51 GMT -- 请求发出的时间
3.3实体内容
只有POST提交的参数会放到实体内容中
四、HttpServletRequest对象
当服务器接收到这些请求信息,怎么处理呢?我们怎么获取呢?
我们Java开发是面向对象的开发,所以服务器把这些数据给我们封装成了一个对象,该对象就是HttpServletRequest对象。
核心的API:
请求行:
request.getMethod(); 请求方式 request.getRequetURI() / request.getRequetURL() 请求资源 request.getProtocol() 请求http协议版本
请求头:
request.getHeader("名称") 根据请求头获取请求值
request.getHeaderNames() 获取所有的请求头名称
实体内容:
request.getInputStream() 获取实体内容数据
四、传递的参数如何获取?
核心的API:
request.getParameter("参数名"); 根据参数名获取参数值(注意,只能获取一个值的参数)
request.getParameterValue("参数名“);根据参数名获取参数值(可以获取多个值的参数)
request.getParameterNames(); 获取所有参数名称列表
关于参数编码问题:
request.setCharacterEncoding("utf-8");
上面的代码,对于中文编码,只能对post方式有效,因为这个设置编码只对实体部的内容有效果。
五、Http响应
5.1、响应信息有三部分组成:响应行,响应头,实体内容
HTTP/1.1 200 OK --响应行
Server: Apache-Coyote/1.1 --响应头(key-vaule)
Content-Length: 24
Date: Fri, 30 Jan 2015 01:54:57 GMT
--一个空行
this is hello servlet!!! --实体内容
5.1.1 响应行
# http 协议版本
#状态码:服务器处理请求的结果(状态)
- 200 : 表示请求处理完成并完美返回
- 302: 表示请求需要进一步细化。
- 404: 表示客户访问的资源找不到。
- 500: 表示服务器的资源发送错误。(服务器内部错误)
#状态描述
5.1.2响应头
Location: http://www.it315.org/index.jsp -表示重定向的地址,该头和302的状态码一起使用。
Server:apache tomcat ---表示服务器的类型
Content-Encoding: gzip -- 表示服务器发送给浏览器的数据压缩类型
Content-Length: 80 --表示服务器发送给浏览器的数据长度
Content-Language: zh-cn --表示服务器支持的语言
Content-Type: text/html; charset=GB2312 --表示服务器发送给浏览器的数据类型及内容编码
Last-Modified: Tue, 11 Jul 2000 18:23:51 GMT --表示服务器资源的最后修改时间
Refresh: 1;url=http://www.it315.org --表示定时刷新
Content-Disposition: attachment; filename=aaa.zip --表示告诉浏览器以下载方式打开资源(下载文件时用到)
Transfer-Encoding: chunked
Set-Cookie:SS=Q0=5Lb_nQ; path=/search --表示服务器发送给浏览器的cookie信息(会话管理用到)
Expires: -1 --表示通知浏览器不进行缓存
Cache-Control: no-cache
Pragma: no-cache
Connection: close/Keep-Alive --表示服务器和浏览器的连接状态。close:关闭连接 keep-alive:保存连接
5.2、HttpServletResponse对象
5.2.1、响应行
response.setStatus() 设置状态码
5.2.2、响应头
response.setHeader("name","value") 设置响应头
5.2.3、实体内容
response.getWriter().writer(); 发送字符实体内容
response.getOutputStream().writer() 发送字节实体内容
5.3、案例分析
5.3.1、请求重定向(相当于超链接跳转页面)
1、浏览器会向服务器发送两次,意味着就有2个request、response
2、用重定向技术,浏览器地址栏会发生变化。
3、用户登录和显示购物车,通常会用到重定向
请求重定向:发送一个状态码302 +location 响应头
response.setStatus(302);
response.setHeader("location", "/ResponseDemo/adv.html");
简化写法
response.sendRedirect("/ResponseDemo/adv.html");
5.3.2、定时刷新
原理:浏览器认识refresh头,得到refresh头之后重新请求资源
response.setHeader("refresh", "1");
隔三秒跳转
response.setHeader("refresh", "3;/ResponseDemo/adv.html");
5.3.3、content-Type
处理不同种类的数据,如XML、html、image等
response.setContentType("image/png");
FileInputStream fileInputStream=new FileInputStream(file);
byte[] buf=new byte[1024];
int len=0;
//把图片内容写到浏览器
while((len=fileInputStream.read(buf))!=-1){
response.getOutputStream().write(buf, 0, len);
}
以下载方式打开
response.setHeader("Content-Disposition","attachment;filename="+file.getName());
如果名字为中文,要记得转码:
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String path=this.getServletContext().getRealPath("/download/小猫.png"); String filename=path.substring(path.lastIndexOf("\\")+1); String filenamedisplay = new String(filename.getBytes("UTF-8"),"iso-8859-1"); response.setHeader("Content-Disposition","attachment;filename=" + filenamedisplay); InputStream in=null; OutputStream out=null; try{ in=new FileInputStream(path); int len=0; byte buffer[]=new byte[1024]; out=response.getOutputStream(); while((len=in.read(buffer))>0){ out.write(buffer,0,len); } }finally{ if(in!=null){ try{ in.close(); }catch(Exception e){ e.printStackTrace(); } } } }
六、Response细节问题
getOutputStream和getWriter方法冲突的问题:
getOutputStream和getWriter方法分别用于得到输出二进制数据、输出文本数据的ServletOuputStream、
Printwriter对象。getOutputStream和getWriter这两个方法互相排斥,调用了其中的任何一个方法后,就不能再调用另一方法。