JAVA基础之HttpServletResponse响应
用户在客户端输入网址(虚拟路径)时,开始发送一个HTTP请求(请求行、请求头、请求体)至服务器。服务器内的Tomcat引擎会解析请求的地址,去找XML文件,然后根据虚拟路径找Servlet的真实路径,真实的Servlet会将请求的信息封装成request(请求)对象,然后再创建一个response(响应)对象,(此时的response内是空的)同时创建servlet对象,并调用service方法(或doGet和doPost方法)。这样就是把两个对象传给了服务器内的某个servlet的service方法,通过这个方法,我们可以获得request的所有的信息,并且向response内设置信息。response.getwriter().write()将内容写到response的缓冲区,这样service方法结束了,方法返回后,tomcat引擎会将从该response缓冲区中获取的设置信息封装成一个HTTP响应(响应行、响应头、响应体),发送给客户端。客户端解析响应回来的东西继而进行显示。
我们可以通过设置修改响应的信息进行相应的重定向(用户访问的网页不存在并跳转到其他网页上)、修改响应文本(需要修改浏览器和服务器两边的编码,并且还得处理兼容问题)。
一、概述:
我们在创建Servlet时会覆盖service()方法,或doGet()/doPost(),这些方法都有两个参数,一个为代表请求的request和代表响应response。
service方法中的response的类型是ServletResponse,而doGet/doPost方法的response的类型是HttpServletResponse,HttpServletResponse是ServletResponse的子接口,功能和方法更加强大
二、运行流程:
三、内容:
响应行、响应头、响应体;
四、通过response 设置响应行:
设置响应行的状态码:setStatus( int sc)
五、通过response 设置响应头:
setHeader(String name,String value) 设置
public class RefreshServlet extends HttpServlet { protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //设置定时刷新的头 response.setHeader("refresh","5;url=https://www.baidu.com"); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> <script type="text/javascript"> window.onload=function(){ //获取span元素 var second=document.getElementById("second"); //定义秒数 var time =5; //设置定时器 var timer=setInterval(function(){ second.innerHTML=time; time--; if(time < 0){ clearInterval(timer); location.href="https://www.baidu.com"; } },1000); } </script> </head> <body> 恭喜您,注册成功! <span id="second" style="color:red">5</span> 秒后跳转,如没跳转,请点击<a href="https://www.baidu.com">这里</a> </body> </html>
1、重定向:(请求服务器两次,地址栏变化)
①、状态码:302;
②、响应头:location 代表重定向地址;
public class Servlet01 extends HttpServlet { private static final long serialVersionUID = 1L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { /*// 设置响应状态码 response.setStatus(302); //设置响应头中的Location response.setHeader("Location","/WEB0/Servlet02");*/ //重定向 response.sendRedirect("/WEB0/Servlet02"); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
public class Servlet02 extends HttpServlet { protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.getWriter().write("Servlet02"); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
六、通过response 设置响应体:
1、响应体设置文本:
PrintWriter
getWriter()
获得字符流,通过字符流的write(
String s)
方法可以将字符串设置到response 缓冲区中,随后Tomcat会将response缓冲区中的内容组装成Http响应返回给浏览 器端。
关于设置中文的乱码问题
原因:response缓冲区的默认编码是iso8859-1,此码表中没有中文,可以通过 response的setCharacterEncoding(
String charset)
设置response的编码,
但我们发现客户端还是不能正常显示文字。
原因:我们将response缓冲区的编码设置成UTF-8,但浏览器的默认编码是本地系统的编码,因为我们都是中文系统,所以客户端浏览器的默认编码是GBK,我们可以手动修改浏览器的编码是UTF-8。
我们还可以在代码中指定浏览器解析页面的编码方式,通过response的setContentType(
String type)
方法指定页面解析时的编码是UTF-8。
response.setContentType("text/html;charset=UTF-8");
上面的代码不仅可以指定浏览器解析页面时的编码,同时也内含 setCharacterEncoding的功能,所以在实际开发中只要编写 response.setContentType("text/html;charset=UTF-8"),就可以解决页面输出中文乱码问题。
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<a href="/WEB0/DownloadServlet?file=乱码.png">乱码.png</a>
<a href="/WEB0/DownloadServlet?file=a.txt">a.txt</a>
<a href="/WEB0/DownloadServlet?file=a.zip">a.zip</a>
</body>
</html>
package com.oracle; import java.io.FileInputStream; import java.io.IOException; import java.net.URLEncoder; import javax.servlet.ServletException; import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import sun.misc.BASE64Encoder; public class DownloadServlet extends HttpServlet { protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //服务器获取文件名(文件名是中文的话,获取的时候就成了乱码了) String filename = request.getParameter("file");//??.png //get请求中---解决文件名中文乱码问题 filename =new String(filename.getBytes("ISO-8859-1"),"UTF-8");//乱码.png //获取User-Agent获取客户端浏览器到底是哪个浏览器 String agent=request.getHeader("User-Agent"); String filenameEncoder=""; if (agent.contains("MSIE")) { // IE浏览器 filenameEncoder= URLEncoder.encode(filename, "utf-8"); filenameEncoder= filenameEncoder.replace("+", " "); } else if (agent.contains("Firefox")) { // 火狐浏览器 BASE64Encoder base64Encoder = new BASE64Encoder(); filenameEncoder= "=?utf-8?B?" + base64Encoder.encode(filename.getBytes("utf-8")) + "?="; } else { // 其它浏览器 filenameEncoder= URLEncoder.encode(filename, "utf-8"); } //告知浏览器文件的类型(响应体) response.setContentType(getServletContext().getMimeType(filename)); //告知浏览器以附件的方式提供下载功能 而不是解析 response.setHeader("Content-Disposition","attachment;filename="+filenameEncoder); //服务器获取后开始进行复制的程序:获取字节输出流 ServletOutputStream sos = response.getOutputStream(); //获取数据源的绝对路径 String realpath = getServletContext().getRealPath("download/"+filename); //获取字节输入流 FileInputStream fis =new FileInputStream(realpath); //开始复制 byte[] bytes=new byte[1024]; int len=0; while((len=fis.read(bytes))!=-1){ sos.write(bytes, 0, len); } //释放资源 fis.close(); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
2、response细节点:
①、response获得的流不需要手动关闭,web容器(tomcat容器)会帮助我们关闭,
②、getWriter和getOutputStream不能同时调用
③、重定向语句一般作为终结代码