用戶在客戶端輸入網址(虛擬路徑)時,開始發送一個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不能同時調用
③、重定向語句一般作為終結代碼