2.3 解决post请求乱码问题 掌握
学习目标
- 能够掌握产生乱码的原因以及如何解决post乱码
内容讲解
html页面:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <form action="/httpServletRequestDemo04Servlet" method="post"> <input type="text" name="username"><br> <input type="submit"> </form> </body> </html>
【1】从tomcat8开始以后,对于get请求乱码,tomcat已经解决。对于post请求中文乱码没有解决,需要我们自己处理。
【2】post请求乱码产生的原因和解决思路
说明:
1)页面使用的编码表是UTF-8编码,tomcat使用的是默认编码表ISO-8859-1进行解码,编码和解码使用的编码表不一致导致乱码。
2)解决思路:先按照ISO-8859-1编码,在按照UTF-8进行重新解码
【3】解决方案
解决方案有三种:
1.方案一
【1】方式一 使用URLEncoder类进行编码:static String encode(String s, String enc) 参数: s:编码的字符串 enc:使用编码表 使用URLDecoder进行解码:static String decode(String s, String enc) 参数: s:解码的字符串 enc:使用编码表
2.方案二
【2】方式二: 使用String类中的方法进行编码: byte[] getBytes(String charsetName) 参数表示指定的编码表,返回值表示编码后的字节数组 使用String类中的构造方法进行解码:String(byte[] bytes, String charsetName) 参数: bytes:字节数组 charsetName:表示指定的编码表 返回值:解码后的字符串
3.方案三
【3】方式三: 如果是get请求,tomcat8底层已经帮助我们解决完了,我们只需要解决post乱码即可,但是上述 两种方式对于post请求可以解决乱码,对于get请求本身获取到的已经是正确的数据,处理 后又乱码了。 我们的想法是:get请求不用我们自己书写代码处理乱码,只需要我们书写代码处理post乱码。 我们接下来学习第三种解决方案: 只解决来自于请求体数据的乱码。而get请求体没有数据,post请求体含有数据,所以我们可以理解为第三种处理方案只 是用来解决post乱码的。使用的api是ServletRequest接口中的: void setCharacterEncoding(String env) 参数:指定的编码表 注意:该方式的代码必须书写在获取请求数据之前
【4】代码实现
import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.net.URLDecoder; import java.net.URLEncoder; @WebServlet("/httpServletRequestDemo04Servlet") public class LoginServlet extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //1.获取浏览器的请求数据 // String username = request.getParameter("username"); /* 解决post乱码问题有三种方式: 【1】方式一 使用URLEncoder类进行编码:static String encode(String s, String enc) 参数: s:编码的字符串 enc:使用编码表 使用URLDecoder进行解码:static String decode(String s, String enc) 参数: s:解码的字符串 enc:使用编码表 */ //1)编码 : 使用URLEncoder类进行编码:static String encode(String s, String enc) // String encodeUsername = URLEncoder.encode(username, "ISO-8859-1"); // //2)解码:使用URLDecoder进行解码:static String decode(String s, String enc) // username = URLDecoder.decode(encodeUsername, "UTF-8"); /* 解决post乱码问题有三种方式: 【2】方式二: 使用String类中的方法进行编码: byte[] getBytes(String charsetName) 参数表示指定的编码表,返回值表示编码后的字节数组 使用String类中的构造方法进行解码:String(byte[] bytes, String charsetName) 参数: bytes:字节数组 charsetName:表示指定的编码表 返回值:解码后的字符串 */ //1)编码 : 使用String类中的方法进行编码: byte[] getBytes(String charsetName) // byte[] bytes = username.getBytes("ISO-8859-1"); // //2)解码:使用String类中的构造方法进行解码:String(byte[] bytes, String charsetName) // username = new String(bytes, "UTF-8"); //username = new String(username.getBytes("ISO-8859-1"), "UTF-8"); /* 解决post乱码问题有三种方式: 【3】方式三: 如果是get请求,tomcat8底层已经帮助我们解决完了,我们只需要解决post乱码即可,但是上述 两种方式对于post请求可以解决乱码,对于get请求本身获取到的已经是正确的数据,处理 后又乱码了。 我们的想法是:get请求不用我们自己书写代码处理乱码,只需要我们书写代码处理post乱码。 我们接下来学习第三种解决方案: 只解决来自于请求体数据的乱码。而get请求体没有数据,post请求体含有数据,所以我们可以理解为第三种处理方案只是用来解决 post乱码的。使用的api是ServletRequest接口中的: void setCharacterEncoding(String env) 参数:指定的编码表 注意:该方式的代码必须书写在获取请求数据之前 */ request.setCharacterEncoding("utf-8");//告知tomcat使用UTF-8解码页面请求数据 // 1.获取浏览器的请求数据 String username = request.getParameter("username"); System.out.println("username = " + username); } }
内容小结
1.tomcat8以后对于get请求乱码已经处理完毕,我们只需要处理post请求
2.处理post请求乱码有三种方式:
方式一:
/* 解决post请求乱码实现: 方式一: 编码:URLEncoder:HTML 格式编码的实用工具类。编码方法: static String encode(String s, String enc) 参数:s 编码的字符串 enc 编码使用的编码表 解码:URLDecoder : HTML 格式解码的实用工具类,解码方法: static String decode(String s, String enc) 参数:s 解码的字符串 enc 解码使用的编码表 */ //编码:URLEncoder:HTML 格式编码的实用工具类。编码方法: // String encode = URLEncoder.encode(username, "ISO-8859-1"); // //解码:URLDecoder : HTML 格式解码的实用工具类,解码方法: // username = URLDecoder.decode(encode, "UTF-8");
方式二:
/* 解决post请求乱码实现: 方式二: 编码:使用String类的方法进行编码:byte[] getBytes(String charsetName) 参数表示指定的码表 解码:使用String类的构造方法:String(byte[] bytes, String charsetName) 参数:第一个参数是字节数组 第二个参数表示指定的码表 */ // 编码:使用String类的方法进行编码:byte[] getBytes(String charsetName) 参数表示指定的码表 // byte[] bytes = username.getBytes("ISO-8859-1"); // // 解码:使用String类的构造方法:String(byte[] bytes, String charsetName) 参数:第一个参数是字节数组 第二个参数表示指定的码表 // username = new String(bytes, "UTF-8"); // username = new String(username.getBytes("ISO-8859-1"), "UTF-8");
方式三:使用最多
/* 解决post请求乱码实现: 方式三:推荐使用 使用request对象调用方法: void setCharacterEncoding(String env) 参数:env指定的编码表 说明: 1.该方法是用来解决请求体数据的乱码问题。get请求体没有数据,post请求体含有数据。可以认为该方法 就是解决post请求乱码的 2.该方法必须放到获取所有请求数据之前处理乱码。 */ //处理请求乱码 request.setCharacterEncoding("utf-8");
2.4 Request请求转发
- 请求转发(forward):一种在服务器内部的资源跳转方式。
(1)浏览器发送请求给服务器,服务器中对应的资源A接收到请求
(2)资源A处理完请求后将请求发给资源B
(3)资源B处理完后将结果响应给浏览器
(4)请求从资源A到资源B的过程就叫请求转发
- 请求转发的实现方式:
req.getRequestDispatcher("资源B路径").forward(req,resp);
具体如何来使用,我们先来看下需求:
针对上述需求,具体的实现步骤为:
1.创建一个RequestDemo5类,接收/req5的请求,在doGet方法中打印demo5
2.创建一个RequestDemo6类,接收/req6的请求,在doGet方法中打印demo6
3.在RequestDemo5的方法中使用
req.getRequestDispatcher(“/req6”).forward(req,resp)进行请求转发
4.启动测试
(1)创建RequestDemo5类
/** * 请求转发 */ @WebServlet("/req5") public class RequestDemo5 extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("demo5..."); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
(2)创建RequestDemo6类
/** * 请求转发 */ @WebServlet("/req6") public class RequestDemo6 extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("demo6..."); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
(3)在RequestDemo5的doGet方法中进行请求转发
/** * 请求转发 */ @WebServlet("/req5") public class RequestDemo5 extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("demo5..."); //请求转发 request.getRequestDispatcher("/req6").forward(request,response); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
(4)启动测试
访问http://localhost:8080/request-demo/req5
,就可以在控制台看到如下内容:
说明请求已经转发到了/req6
- 请求转发资源间共享数据:使用Request对象
此处主要解决的问题是把请求从/req5
转发到/req6
的时候,如何传递数据给/req6
。
需要使用request对象提供的三个方法:
- 存储数据到request域[范围,数据是存储在request对象]中
void setAttribute(String name,Object o);
- 根据key获取值
Object getAttribute(String name);
- 根据key删除该键值对
void removeAttribute(String name);
接着上个需求来:
1.在RequestDemo5的doGet方法中转发请求之前,将数据存入request域对象中
2.在RequestDemo6的doGet方法从request域对象中获取数据,并将数据打印到控制台
3.启动访问测试
(1)修改RequestDemo5中的方法
@WebServlet("/req5") public class RequestDemo5 extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("demo5..."); //存储数据 request.setAttribute("msg","hello"); //请求转发 request.getRequestDispatcher("/req6").forward(request,response); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
(2)修改RequestDemo6中的方法
/** * 请求转发 */ @WebServlet("/req6") public class RequestDemo6 extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("demo6..."); //获取数据 Object msg = request.getAttribute("msg"); System.out.println(msg); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
(3)启动测试
访问http://localhost:8080/request-demo/req5
,就可以在控制台看到如下内容:
此时就可以实现在转发多个资源之间共享数据。
- 请求转发的特点
- 浏览器地址栏路径不发生变化
虽然后台从/req5
转发到/req6
,但是浏览器的地址一直是/req5
,未发生变化
- 只能转发到当前服务器的内部资源
不能从一个服务器通过转发访问另一台服务器
- 一次请求,可以在转发资源间使用request共享数据
虽然后台从/req5
转发到/req6
,但是这个只有一次请求
3.HTTP响应详解(理解)
1.使用抓包查看响应报文协议内容
学习目标
- 能够使用抓包查看响应报文协议内容
内容讲解
注意:
http响应报文协议包括:
1.响应行 2.响应头 3.响应体
响应数据:是服务器响应给浏览器
【1】步骤
1.创建html页面 2.创建servlet
【2】实现
1.创建html页面
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Insert title here</title> </head> <body> <h2>get请求</h2> <form action="/getServlet" method="get"> 用户名:<input type="text" name="username" value="suoge" /> <br/> 密码:<input type="text" name="password" value="1234" /> <br/> <input type="submit" value="get提交" /> </form> <h2>post请求</h2> <form action="/postServlet" method="post"> 用户名:<input type="text" name="username" value="suoge" /> <br/> 密码:<input type="text" name="password" value="1234" /> <br/> <input type="submit" value="post提交" /> </form> </body> </html>
2.创建servlet
import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; @WebServlet("/getServlet") public class GetServlet extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //响应给浏览器数据 response.getWriter().print("get...."); } }
package com.itheima.sh.a_http_01; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; @WebServlet("/postServlet") public class PostServlet extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //响应给浏览器数据 response.getWriter().print("post...."); } }
【3】抓包结果
内容小结
1.由于浏览器的原因,浏览器会把请求行和响应行信息放在了一起;
2.get和post请求的响应没有区别;
2.HTTP响应报文协议介绍
学习目标
- 理解响应报文协议的组成部分
内容讲解
【1】响应行
响应行格式:协议/版本 状态码
如:HTTP/1.1 200 ;
状态码 | 状态码描述 | 说明 |
200 | OK | 请求已成功,请求所希望的响应头或数据体将随此响应返回。出现此状态码是表示正常状态。 |
302 | Move temporarily | 重定向,请求的资源临时从不同的 地址响应请求。 |
304 | Not Modified | 从浏览器缓存中读取数据,不从服务器重新获取数据。例如,用户第一次从浏览器访问服务器端图片资源,以后在访问该图片资源的时候就不会再从服务器上加载而直接到浏览器缓存中加载,这样效率更高。 |
404 | Not Found | 请求资源不存在。通常是用户路径编写错误,也可能是服务器资源已删除。 |
403 | Forbidden | 服务器已经理解请求,但是拒绝执行它 |
405 | Method Not Allowed | 请求行中指定的请求方法不能被用于请求相应的资源 |
500 | Internal Server Error | 服务器内部错误。通常程序抛异常 |
【2】响应头
响应头也是用的键值对key:value,服务器基于响应头通知浏览器的行为。
常见的响应头 :
响应头Key | 响应头value |
location | 指定响应的路径,需要与状态码302配合使用,完成重定向 |
content-Type | 响应正文的类型(MIME类型,属于服务器里面的一种类型,例如文件在window系统有自己的类型,.txt .doc .jpg。文件在服务器中也有自己的类型),同时还可以解决乱码问题。例如:text/html;charset=UTF-8 |
content-disposition | 通过浏览器以附件形式解析正文,例如:attachment;filename=xx.zip |
refresh | 页面刷新,例如:3;url=www.itcast.cn //三秒刷新页面到www.itcast.cn |
常见的MIME类型:就是文件在tomcat服务器中的文件类型:
windows tomcat(MIME类型) 超文本标记语言文本 .html text/html *** xml文档 .xml text/xml XHTML文档 .xhtml application/xhtml+xml 普通文本 .txt text/plain *** PDF文档 .pdf application/pdf Microsoft Word文件 .word application/msword PNG图像 .png image/png ** GIF图形 .gif image/gif JPEG图形 .jpeg,.jpg image/jpeg ** ......
【3】响应体
响应体,就是服务器发送给浏览器的数据。当前浏览器向服务器请求的资源是hello.html,所以服务器给浏览器响应的数据是一个html页面。
请求资源路径:
响应结果:
如果请求是servlet,那么浏览器的响应体接收到的是servlet响应的数据:
内容小结
1.响应行:
协议版本号 状态码 200(一切正常) 404(找不到资源路径) 500(服务器报异常) 302(和location一起使用,实现重定向) 304(从浏览器缓存中读取数据) 405(服务器的servlet没有重写doGet和doPost方法)
2.响应头:
location 指定响应的路径
content-type:告诉浏览器文件格式,告诉浏览器不要解析html文件(text/plain),解决中文乱码问题
refresh 定时刷新
content-disposition 以附件形式展示图片等资源
3.响应体:
服务器处理的结果响应到浏览器中
4.Response对象
1 Response对象介绍
前面讲解完Request对象,接下来我们回到刚开始的那张图:
- Request:使用request对象来获取请求数据
- Response:使用response对象来设置响应数据
Reponse的继承体系和Request的继承体系也非常相似:
介绍完Response的相关体系结构后,接下来对于Response我们需要学习如下内容:
- Response设置响应数据的功能介绍
- Response完成重定向
- Response响应字符数据
- Response响应字节数据
2 Response设置响应数据功能介绍
HTTP响应数据总共分为三部分内容,分别是响应行、响应头、响应体,对于这三部分内容的数据,respone对象都提供了哪些方法来进行设置?
- 响应行
对于响应行,比较常用的就是设置响应状态码:
void setStatus(int sc);
- 响应头
设置响应头键值对:
void setHeader(String name,String value);
- 响应体
对于响应体,是通过字符、字节输出流的方式往浏览器写,
获取字符输出流:
PrintWriter getWriter();
获取字节输出流
ServletOutputStream getOutputStream();
介绍完这些方法后,后面我们会通过案例把这些方法都用一用,首先先来完成下重定向的功能开发。