javaWeb學習日記_17:request請求
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來解碼。