1. request功能介紹
- 獲取請求頭
- 獲取請求參數
- Servlet三大域對象之一
- 請求包含和請求轉發
2. request域方法
- void setAttribute(String name, Object value):添加或替換request域屬性
- Object getAttribute(String name):獲取request域指定名稱的域屬性
- void removeAttribute(String name):移除request域指定名稱的域屬性
- Enumeration getAttributeNames():獲取所有request域的屬性名稱
3. request獲取請求頭
- String getHeader(String name):獲取指定名稱的請求頭
- int getIntHeader(String name):獲取指定名稱的請求頭,把值轉換成int類型。
- Enumeration getHeaderNames():獲取所有請求頭名稱
4. request請求數據相關其他方法
重點:
- String getMethod():獲取請求方式
- String getContextPath():獲取上下文路徑,即“/” + 應用名稱,例如:/day05_1
- void setCharacterEncoding(String):設置請求體的編碼
- String getRemoteAddr():獲取客戶端IP地址
package cn.itcast.servlet;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class CServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
/*
* 通過request來獲取url的相關方法
*/
response.getWriter().print(request.getScheme() + "<br/>");//獲取請求協議
response.getWriter().print(request.getServerName() + "<br/>");//獲取服務器名稱
response.getWriter().print(request.getServerPort() + "<br/>");//獲取服務器端口號
response.getWriter().print(request.getContextPath() + "<br/>");//獲取項目名稱
response.getWriter().print(request.getServletPath() + "<br/>");//獲取Servlet路徑
response.getWriter().print(request.getQueryString() + "<br/>");//獲取參數部分
response.getWriter().print(request.getRequestURI() + "<br/>");//獲取請求URI
response.getWriter().print(request.getRequestURL() + "<br/>");//獲取請求URL
}
}
非重點:
- int getContentLength():獲取請求體字節數
- Locale getLocale():獲取請求Locale,例如zh_CN表示中文,中國
- String getCharacterEncoding():獲取請求體編碼,在沒有調用setCharacterEncoding()之前該方法返回null
- String getQueryString():獲取參數列表,例如:username=zhangSan&password=123
- String getRequestURI():返回請求URI路徑,從應用名稱開始,到參數之前這一段,例如:/day05_1/AServlet
- StringBuffer getRequestURL():整個請求URL,不包含參數部分
- String getServletPath():返回Servlet路徑,從應用名稱後開始,到參數之前這一段,不包含應用名稱。
- String getServerName():返回主機名,例如:localhost
- int getServerPort():返回服務器端口號,例如:8080
5. 請求參數
- 獲取請求參數,即獲取超鏈接上的參數和表單中的參數
- String getParameter(String name):獲取指定名稱的參數,如果存在同名參數,那麼該方法只獲取第一個參數值
- String[] getParameterValues(String name):獲取指定名稱的參數,因爲同名參數的存在,所以返回值爲String[]
- Enumeration getParameterNames():獲取所有參數名稱
- Map getParameterMap():獲取所有參數,封裝到Map中,key爲參數名稱,value爲參數值。
AServlet.java內代碼:
import java.io.IOException;
import java.util.Arrays;
import java.util.Enumeration;
import java.util.Map;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 演示request獲取請求參數!
* @author cxf
*
*/
public class AServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("GET: " + request.getParameter("xxx"));
System.out.println("GET: " + request.getParameter("yyy"));
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
String username = request.getParameter("username");
String password = request.getParameter("password");
String[] hobby = request.getParameterValues("hobby");
System.out.println(username + ", " + password + ", " + Arrays.toString(hobby));
/*
* 測試獲取所有請求參數的名稱
*/
Enumeration names = request.getParameterNames();
while(names.hasMoreElements()) {
System.out.println(names.nextElement());
}
/*
* 獲取所有請求參數,封裝到Map中
*/
Map<String,String[]> map = request.getParameterMap();
for(String name : map.keySet()) {
String[] values = map.get(name);
System.out.println(name + "=" + Arrays.toString(values));
}
}
}
表單代碼:a.htmml
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN">
<html>
<head>
<title>a.html</title>
<meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
<meta http-equiv="description" content="this is my page">
<meta http-equiv="content-type" content="text/html; charset=UTF-8">
<!--<link rel="stylesheet" type="text/css" href="./styles.css">-->
</head>
<body>
<h1>測試請求參數</h1>
<a href="/day10_3/AServlet?xxx=XXX&yyy=YYY">點擊這裏</a>
<hr/>
<form action="/day10_3/AServlet" method="post">
用戶名:<input type="text" name="username"/><br/>
密 碼:<input type="password" name="password"/><br/>
愛 好:<input type="checkbox" name="hobby" value="cf"/>吃飯
<input type="checkbox" name="hobby" value="sj"/>睡覺
<input type="checkbox" name="hobby" value="ddm"/>打代碼
<br/>
<input type="submit" value="提交"/>
</form>
</body>
</html>
6. 請求包含和請求轉發
- 請求包含和請求轉發都是在一個請求中,訪問兩個Servlet。
- 請求包含和請求轉發都是有一個Servlet去調用執行另一個Servlet
- 請求包含和請求轉發都可以共享request中的數據,因爲都是一個請求。
* 從AServlet請求轉發到BServlet
> 在AServlet中可以設置響應頭
> 在AServlet中不能使用響應流輸出
如果在AServlet中執行了響應操作,那麼有兩種可能:第一種可能:如果在AServlet中響應的數據導致response提交,那麼在轉發時拋出異常;第二種可能:如果在AServlet中響應的數據沒有導致response提交,那麼response中的數據會被清空。
從AServlet請求包含BServlet:在AServlet可以設置響應頭,在AServlet可以使用響應流輸出
- 請求轉發和請求包含都要使用RequestDispatcher對象:RequestDispatcher rd = request.getRequestDispatcher("/BServlet");
- 請求轉發執行RequestDispatcher的forward()方法:rd.forward(request,response);
- 請求包含執行RequestDispatcher的include()方法:rd.include(request,response);
- 請求轉發和請求包含的路徑都是服務器端路徑,相對當前應用
轉發演示:
OneServlet.java內代碼:
package cn.itcast.servlet.forward;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 演示請求轉發
* @author cxf
*
*/
public class OneServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("OneServlet...");
response.setHeader("aaa", "AAA");//設置響應頭
// for(int i = 0; i < 1024 * 24 + 1; i++) {
// response.getWriter().print("a");//設置響應體
// }
/*
* 向reuqest域中添加一個屬性
*/
request.setAttribute("username", "zhangsan");
// 等同與調用TwoServlet的service()方法。
request.getRequestDispatcher("/TwoServlet").forward(request, response);//請求轉發
}
}
TwoServlet.java內代碼:
package cn.itcast.servlet.forward;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class TwoServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println(request.getAttribute("username"));
System.out.println("TwoServlet...");
response.getWriter().print("hello TwoServlet!");//設置響應體
}
}
包含演示:
OneServlet.java內代碼:
package cn.itcast.servlet.include;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class OneServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("OneServlet...");
response.setHeader("aaa", "AAA");//設置響應頭
response.getWriter().print("aaaa");//設置響應體
request.getRequestDispatcher("/TwoServlet").include(request, response);//請求轉發
}
}
TwoServlet.java內代碼:
package cn.itcast.servlet.include;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class TwoServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("TwoServlet...");
response.getWriter().print("hello TwoServlet!");//設置響應體
}
}
7. 請求轉發與重定向
- 請求轉發是一個請求,而重定向是兩個請求
- 請求轉發,是使用RequestDispatcher來完成,重定向使用response對象來完成
- 請求轉發的路徑都是服務器端路徑,而重定向是客戶端路徑,需要給出應用名稱
- 請求轉發在瀏覽器地址欄中的地址是第一個Servlet的路徑,而重定向在地址欄中的地址是第二個請求的Servlet的路徑
- 請求轉發中的兩個Servlet是可以共享request數據的,而重定向因爲是兩個請求,所以不能共享request數據
- 請求轉發只能轉發到本應用的其他Servlet,而重定向可以重定向到其他應用中。
8. request.getParameter()和request.getAttribute()
- getParameter()是獲取客戶端參數,它是從客戶端傳遞給服務器的數據。
- getAttribute()是獲取服務器端自己設置的數據,而不是客戶端的數據。
- request沒有setParameter()方法,不能自己設置參數,參數都由客戶端傳遞
- request有setAttribute()方法,在getAttribute()之前,需要先setAttribute()才能獲取到。
- getAttribute()和setAttribute()是用來在請求轉發和請求包含中的多個Servlet中共享數據。
補充內容 :
路徑
1. 客戶端路徑和服務器端路徑
- 客戶端路徑需要給出應用名稱,例如:/day05_1/AServlet
- 服務器端路徑無需給出應用名稱,例如:/AServlet
2. 客戶端路徑
1). 頁面中都是客戶端路徑:
* 超鏈接的href
* 表單的action
* <img>的src
2). 重定向也是客戶端路徑:response.sendRedirect("/day05_1/BServlet");
3. 服務器端路徑
* <url-pattern>
* 請求轉發和請求包含
* ServletContext獲取資源等
亂碼
1. 請求編碼
* 客戶端發送的數據編碼:由瀏覽器來決定:
1). 如果是在地址欄中直接給出url,那麼一般都是默認爲GBK,但這個可能不太大。
2). 如果是通過頁面上的表單或超鏈接發出請求,那麼由當前頁面的編碼來決定發送的參數的編碼。
* 無論瀏覽器發送過來的是什麼編碼的數據,Tomcat都默認使用ISO-8859-1來解碼
1). POST:可以使用request.setCharacterEncoding()方法來設置請求體數據的編碼,因爲POST請求參數在請求體中,所以是可以設置編碼的。在使用request.getParameter()方法獲取參數之前,先使用request.setCharacterEncoding()方法來設置編碼即可。
2). GET:沒有方法可以設置它,因爲參數在url中。所以使用request.getParameter()獲取到的數據一定是錯誤的使用了iso-8859-1解碼的。可以再使用iso-8859-1把字符串轉回到byte[],再重新使用正確的編碼來解碼即可。
String s = request.getParameter("s");//使用iso-8859-1錯誤的解碼了
byte[] bytes = s.getBytes("iso-8859-1");//退回錯誤的解碼,讓字符串通過iso-8859-1返回到字節數據,即還原字節數據
s = new String(bytes, "utf-8");//重新使用正確的utf-8來解碼。