1. 程式人生 > >javaWeb學習日記_17:request請求

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來解碼。