Http协议入门
2016-03-31 11:21
465 查看
什么是http协议
http协议: 对浏览器客户端 和 服务器端 之间数据传输的格式规范
查看http协议的工具
1)使用火狐的firebug插件(右键->firebug->网络) 2)使用谷歌的“审查元素” 3)使用系统自带的telnet工具(远程访问工具) a)telnet localhost 8080 访问tomcat服务器 b)ctrl+] 回车 可以看到回显 c)输入请求内容
GET /day09/hello HTTP/1.1 Host: localhost:8080
d)回车,即可查看到服务器响应信息。
http协议内容
请求(浏览器 -> 服务器)GET /day09/hello HTTP/1.1 Host: localhost:8080User-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
响应(服务器 -> 浏览器)
HTTP/1.1 200 OK Server: Apache-Coyote/1.1 Content-Length: 24 Date: Fri, 30 Jan 2015 01:54:57 GMT this is hello servlet!!!
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 -- (可选)实体内容
请求行
GET /day09/hello HTTP/1.1
http协议版本
http1.0:当前浏览器客户端与服务器端建立连接之后,只能发送一次请求,一次请求之后连接关闭。 http1.1:当前浏览器客户端与服务器端建立连接之后,可以在一次连接中发送多次请求。(基本都使用1.1)
请求资源
URL: 统一资源定位符。http://localhost:8080/day09/testImg.html。只能定位互联网资源。是URI的子集。 URI: 统一资源标记符。/day09/hello。用于标记任何资源。可以是本地文件系统,局域网的资源(//192.168.14.10/myweb/index.html),可以是互联网。
请求方式
常见的请求方式: GET 、 POST、 HEAD、 TRACE、 PUT、 CONNECT 、DELETE 常用的请求方式: GET 和 POST 表单提交: <form action="提交地址" method="GET/POST"> <form>
GET vs POST 区别
1)GET方式提交 a)地址栏(URI)会跟上参数数据。以?开头,多个参数之间以&分割。
GET /day09/testMethod.html?name=eric&password=123456 HTTP/1.1 Host: localhost:8080 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 Referer: http://localhost:8080/day09/testMethod.html Connection: keep-alive
b)GET提交参数数据有限制,不超过1KB。 c)GET方式不适合提交敏感密码。 d)注意: 浏览器直接访问的请求,默认提交方式是GET方式 2)POST方式提交 a)参数不会跟着URI后面。参数而是跟在请求的实体内容中。没有?开头,多个参数之间以&分割。
POST /day09/testMethod.html HTTP/1.1 Host: localhost:8080 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 Referer: http://localhost:8080/day09/testMethod.html Connection: keep-alive name=eric&password=123456
b)POST提交的参数数据没有限制。 c)POST方式提交敏感数据。
请求头
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 -- 请求发出的时间
实体内容
只有POST提交的参数会放到实体内容中
HttpServletRequest对象
HttpServletRequest对象作用是用于获取请求数据。 核心的API: 请求行: request.getMethod(); 请求方式 request.getRequetURI() / request.getRequetURL() 请求资源 request.getProtocol() 请求http协议版本 请求头: request.getHeader("名称") 根据请求头获取请求值 request.getHeaderNames() 获取所有的请求头名称 实体内容: request.getInputStream() 获取实体内容数据
package gz.itcast.b_request; import java.io.IOException; import java.io.InputStream; import java.util.Enumeration; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 请求数据的获取 * @author APPle * */ public class RequestDemo1 extends HttpServlet { /** * 1)tomcat服务器接收到浏览器发送的请求数据,然后封装到HttpServetRequest对象 * 2)tomcat服务器调用doGet方法,然后把request对象传入到servlet中。 */ public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { /** * 3)从request对象取出请求数据。 */ //t1(request); //t2(request); } // 为了接收POST方式提交的请求 @Override protected void doPost(HttpServletRequest request, HttpServletResponse resp) throws ServletException, IOException { /** * 3.3 请求的实体内容 */ InputStream in = request.getInputStream(); //得到实体内容 byte[] buf = new byte[1024]; int len = 0; while( (len=in.read(buf))!=-1 ){ String str = new String(buf,0,len); System.out.println(str); } } private void t2(HttpServletRequest request) { /** * 3.2 请求头 */ String host = request.getHeader("Host"); //根据头名称的到头的内容 System.out.println(host); //遍历所有请求头 Enumeration<String> enums = request.getHeaderNames(); //得到所有的请求头名称列表 while(enums.hasMoreElements()){//判断是否有下一个元素 String headerName = enums.nextElement(); //取出下一个元素 String headerValue = request.getHeader(headerName); System.out.println(headerName+":"+headerValue); } } private void t1(HttpServletRequest request) { /** * 3.1 请求行 格式:(GET /day09/hello HTTP/1.1) */ System.out.println("请求方式:"+request.getMethod());//请求方式 System.out.println("URI:"+request.getRequestURI());//请求资源 System.out.println("URL:"+request.getRequestURL()); System.out.println("http协议版本:"+request.getProtocol());//http协议 } }
service 和 doXX方法区别
注意:tomcat服务器首先会调用servlet的service方法,然后在service方法中再根据请求方式来分别调用对应的doXX方法
package gz.itcast.b_request; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class RequestDemo2 extends HttpServlet { /** * 注意:tomcat服务器首先会调用servlet的service方法,然后在service方法中再根据请求方式来分别调用对应的doXX方法 * (例如,如果是GET请求方式,在service方法中调用doGet方法) * * 因为最常的请求方式是GET 和POST,所以编写servlet程序,只需要覆盖doGet和doPost即可!!!! */ /*@Override protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println(req.getMethod()); System.out.println("service方法被调用"); }*/ /** * 该方法用于接收浏览器的Get方式提交的请求 */ public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("GET方式提交"); } /** * 该方法用于接收浏览器的Post方式提交的请求 */ public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("Post方式提交"); } }
案例-获取浏览器的类型(user-agent)
package gz.itcast.b_request; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 案例-获取浏览器的类型 * @author APPle * */ public class RequestDemo3 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.setContentType("text/html;charset=utf-8"); //获取请求头: user-agent String userAgent = request.getHeader("user-agent"); System.out.println(userAgent); //判断用户使用的浏览器类型 if(userAgent.contains("Firefox")){ response.getWriter().write("你正在使用火狐浏览器"); }else if(userAgent.contains("Chrome")){ response.getWriter().write("你正在使用谷歌浏览器"); }else if(userAgent.contains("Trident")){ response.getWriter().write("你正在使用IE浏览器"); }else{ response.getWriter().write("地球上没有这个浏览器,建议使用火狐浏览器"); } } }
案例- 防止非法链接(referer)
package gz.itcast.b_request; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 案例- 防止非法链接 * 这是需要下载的资源 * @author APPle * */ public class RequestDemo4 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.setContentType("text/html;charset=utf-8"); //得到referer头 String referer = request.getHeader("referer"); System.out.println("referer="+referer); /** * 判断非法链接: * 1)直接访问的话referer=null * 2)如果当前请求不是来自广告 */ if(referer==null || !referer.contains("/day09/adv.html")){ response.getWriter().write("当前是非法链接,请回到首页。<a href='/day09/adv.html'>首页</a>"); }else{ //正确的链接 response.getWriter().write("资源正在下载..."); } } }
传递的请求参数如何获取
GET方式: 参数放在URI后面 POST方式: 参数放在实体内容中 获取GET方式参数: request.getQueryString(); 获取POST方式参数: request.getInputStream(); 问题:但是以上两种不通用,而且获取到的参数还需要进一步地解析。 所以可以使用统一方便的获取参数的方式: 核心的API: request.getParameter("参数名"); 根据参数名获取参数值(注意,只能获取一个值的参数) request.getParameterValue("参数名“);根据参数名获取参数值(可以获取多个值的参数) request.getParameterNames(); 获取所有参数名称列表
请求参数编码问题
修改POST方式参数编码: request.setCharacterEncoding("utf-8"); 修改GET方式参数编码: 手动解码:String name = new String(name.getBytes("iso-8859-1"),"utf-8");
package gz.itcast.b_request; import java.io.IOException; import java.io.InputStream; import java.util.Enumeration; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 获取GET方式和Post方式提交的参数 * @author APPle * */ public class RequestDemo5 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { /** * 设置参数查询的编码 * 该方法只能对请求实体内容的数据编码起作用。POST提交的数据在实体内容中,所以该方法对POST方法有效! * GET方法的参数放在URI后面,所以对GET方式无效!!! */ request.setCharacterEncoding("utf-8"); /* System.out.println("GET方式"); //接收GET方式提交的参数 String value = request.getQueryString(); System.out.println(value);*/ /** * 统一方便地获取请求参数的方法 */ System.out.println(request.getMethod()+"方式"); //getParameter(name): 根据参数名得到参数值(只能获取一个值的参数) String name = request.getParameter("name"); /** * 手动重新解码(iso-8859-1 字符串-> utf-8 字符串) */ if("GET".equals(request.getMethod())){ name = new String(name.getBytes("iso-8859-1"),"utf-8"); } String password = request.getParameter("password"); if("GET".equals(request.getMethod())){ password = new String(password.getBytes("iso-8859-1"),"utf-8"); } System.out.println(name+"="+password); System.out.println("============================="); Enumeration<String> enums = request.getParameterNames(); while( enums.hasMoreElements() ){ String paramName = enums.nextElement(); //如果参数名是hobit,则调用getParameterValues if("hobit".equals(paramName)){ /** * getParameterValues(name): 根据参数名获取参数值(可以获取多个值的同名参数) */ System.out.println(paramName+":"); String[] hobits = request.getParameterValues("hobit"); for(String h: hobits){ if("GET".equals(request.getMethod())){ h = new String(h.getBytes("iso-8859-1"),"utf-8"); } System.out.print(h+","); } System.out.println(); //如果不是hobit,则调用getParameter }else{ String paramValue = request.getParameter(paramName); if("GET".equals(request.getMethod())){ paramValue = new String(paramValue.getBytes("iso-8859-1"),"utf-8"); } System.out.println(paramName+"="+paramValue); } } } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { /*System.out.println("POST方式"); InputStream in = request.getInputStream(); byte[] buf = new byte[1024]; int len = 0; while( (len=in.read(buf))!=-1 ){ System.out.println(new String(buf,0,len)); }*/ /** * 统一方便地获取请求参数的方法 */ /*System.out.println("POST方式"); //根据参数名得到参数值 String name = request.getParameter("name"); String password = request.getParameter("password"); System.out.println(name+"="+password); System.out.println("============================="); Enumeration<String> enums = request.getParameterNames(); while( enums.hasMoreElements() ){ String paramName = enums.nextElement(); String paramValue = request.getParameter(paramName); System.out.println(paramName+"="+paramValue); }*/ //一定调用doGet方式 this.doGet(request, response); } }
Http响应
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!!! --实体内容
响应行
HTTP/1.1 200 OK
http协议版本
状态码
服务器处理请求的结果(状态) 100~199 表示成功接收请求,要求客户端继续提交下一次请求才能完成整个处理过程 200~299 表示成功接收请求并已完成整个处理过程,常用200 300~399 为完成请求,客户需进一步细化请求。例如,请求的资源已经移动一个新地址, 常用302、307和304 400~499 客户端的请求有错误,常用404 500~599 服务器端出现错误,常用 500 常见的状态: 200 : 表示请求处理完成并完美返回 302: 表示请求需要进一步细化。 404: 表示客户访问的资源找不到。 500: 表示服务器的资源发送错误。(服务器内部错误)
状态描述
常见的响应头
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:保存连接
HttpServletResponse对象
HttpServletResponse对象修改响应信息: 响应行: response.setStatus() 设置状态码 响应头: response.setHeader("name","value") 设置响应头 实体内容: response.getWriter().write(); 发送字符实体内容 response.getOutputStream().write() 发送字节实体内容
package gz.itcast.c_response; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 设置响应信息 * @author APPle * */ public class ResponseDemo1 extends HttpServlet { /** * 1)tomcat服务器把请求信息封装到HttpServletRequest对象,且把响应信息封装到HttpServletResponse * 2)tomcat服务器调用doGet方法,传入request,和response对象 */ public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { /** * 3)通过response对象改变响应信息 */ /** * 3.1 响应行 */ //response.setStatus(404);//修改状态码 //response.sendError(404); // 发送404的状态码+404的错误页面 /** * 3.2 响应头 */ response.setHeader("server", "JBoss"); /** * 3.3 实体内容(浏览器直接能够看到的内容就是实体内容) */ //response.getWriter().write("01.hello world"); //字符内容。 response.getOutputStream().write("02.hello world".getBytes());//字节内容 } /** * 4)tomcat服务器把response对象的内容转换成响应格式内容,再发送给浏览器解析。 */ }
案例- 请求重定向(Location)
package gz.itcast.c_response; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 案例- 请求重定向 * (相当于超链接跳转页面) * @author APPle * */ public class ResponseDemo2 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { /** * 需求: 跳转到adv.html * 使用请求重定向: 发送一个302状态码+location的响应头 */ /*response.setStatus(302);//发送一个302状态码 response.setHeader("location", "/day09/adv.html"); //location的响应头 */ //请求重定向简化写法 response.sendRedirect("/day09/adv.html"); } }
案例- 定时刷新(refresh)
package gz.itcast.c_response; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 案例- 定时刷新 * @author APPle * */ public class ResponseDemo3 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { /** * 定时刷新 * 原理:浏览器认识refresh头,得到refresh头之后重新请求当前资源 */ //response.setHeader("refresh", "1"); //每隔1秒刷新次页面 /** * 隔n秒之后跳转另外的资源 */ response.setHeader("refresh", "3;url=/day09/adv.html");//隔3秒之后跳转到adv.html } }
案例-content-Type作用
package gz.itcast.c_response; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * 案例- content-Type作用 * @author APPle * */ public class ResponseDemo4 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { /** * 设置响应实体内容编码 */ //response.setCharacterEncoding("utf-8"); /** * 1. 服务器发送给浏览器的数据类型和内容编码 */ //response.setHeader("content-type", "text/html"); response.setContentType("text/html;charset=utf-8");//和上面代码等价。推荐使用此方法 //response.setContentType("text/xml"); //response.setContentType("image/jpg"); //response.getWriter().write("<html><head><title>this is tilte</title></head><body>中国</body></html>"); response.getOutputStream().write("<html><head><title>this is tilte</title></head><body>中国</body></html>".getBytes("utf-8")); File file = new File("e:/mm.jpg"); /** * 设置以下载方式打开文件 */ response.setHeader("Content-Disposition", "attachment; filename="+file.getName()); /** * 下载图片 */ /** * 发送图片 */ FileInputStream in = new FileInputStream(file); byte[] buf = new byte[1024]; int len = 0; //把图片内容写出到浏览器 while( (len=in.read(buf))!=-1 ){ response.getOutputStream().write(buf, 0, len); }*/ } }
项目中的编码问题
相关文章推荐
- TCP/IP、Http、Socket的区别
- Postman(网页调试)
- TCP连接的建立与关闭
- HTTP POST GET 本质区别详解
- 基于kafka的大量httpRequest队列控制及mergeRequest机制
- tcp服务器端,客户端练习程序
- TCP/IP协议族的四个层次及不同层次的协议
- 基于 openssl + vs2013 https客户端 Demo
- 利用http协议实现图片窃取
- Universal-Image-Loader,android-Volley,Picasso、Fresco和Glide五大Android开源组件加载网络图片的优缺点比较
- java HTP协议之HttpConnUtil
- 批命令连接指定wifi 网络
- 【译】在ASP.Net和IIS中删除不必要的HTTP响应头
- Flash加载网络图片显示
- bzoj3597方伯伯运椰子
- Httpclient POST
- 安卓监听网络状态改变
- Apache搭建http协议版本库
- Runtime初涉之消息转发 http://www.cocoachina.com/ios/20151015/13769.html
- VirtualBox虚拟机网络设置