JavaWeb:HttpServletResponse和HttpServletRequest
請求響應(yīng)流程圖
1. HttpServletResponse
1.1 Response概述
Response是Servlet.service方法的一個(gè)參數(shù),類型為javax.servlet.http.HttpServletResponse。在客戶端發(fā)出每個(gè)請求時(shí),服務(wù)器都會創(chuàng)建一個(gè)response對象,并傳入給Servlet.service()方法。response對象是用來對客戶端進(jìn)行響應(yīng)的,這說明在service()方法中使用response對象可以完成對客戶端的響應(yīng)工作
response對象的功能分為以下四種:
- 設(shè)置響應(yīng)頭信息
- 發(fā)送狀態(tài)碼
- 設(shè)置響應(yīng)正文
- 重定向
1.2 response響應(yīng)正文
response是響應(yīng)對象,向客戶端輸出響應(yīng)正文(響應(yīng)體)可以使用response的響應(yīng)流,repsonse一共提供了兩個(gè)響應(yīng)流對象:
PrintWriter out = response.getWriter();//獲取字符流 ServletOutputStream out = response.getOutputStream();//獲取字節(jié)流當(dāng)然,如果響應(yīng)正文內(nèi)容為字符,那么使用response.getWriter(),如果響應(yīng)內(nèi)容是字節(jié),例如下載時(shí),那么可以使用response.getOutputStream()
注意,在一個(gè)請求中,不能同時(shí)使用這兩個(gè)流!也就是說,要么你使用repsonse.getWriter(),要么使用response.getOutputStream(),但不能同時(shí)使用這兩個(gè)流。不然會拋出IllegalStateException異常
1.2.1 字符響應(yīng)流
- 字符編碼
在使用response.getWriter()時(shí)需要注意默認(rèn)字符編碼為ISO-8859-1,如果希望設(shè)置字符流的字符編碼為utf-8,可以使用response.setCharaceterEncoding(“utf-8”)來設(shè)置。這樣可以保證輸出給客戶端的字符都是使用UTF-8編碼的!
但客戶端瀏覽器并不知道響應(yīng)數(shù)據(jù)是什么編碼的!如果希望通知客戶端使用UTF-8來解讀響應(yīng)數(shù)據(jù),那么還是使用response.setContentType(“text/html;charset=utf-8”)方法比較好,因?yàn)檫@個(gè)方法不只會調(diào)用response.setCharaceterEncoding(“utf-8”),還會設(shè)置content-type響應(yīng)頭,客戶端瀏覽器會使用content-type頭來解讀響應(yīng)數(shù)據(jù)。
- 緩沖區(qū)
response.getWriter()是PrintWriter類型,所以它有緩沖區(qū),緩沖區(qū)的默認(rèn)大小為8KB。也就是說,在響應(yīng)數(shù)據(jù)沒有輸出8KB之前,數(shù)據(jù)都是存放在緩沖區(qū)中,而不會立刻發(fā)送到客戶端。當(dāng)Servlet執(zhí)行結(jié)束后,服務(wù)器才會去刷新流,使緩沖區(qū)中的數(shù)據(jù)發(fā)送到客戶端。
如果希望響應(yīng)數(shù)據(jù)馬上發(fā)送給客戶端:
- 向流中寫入大于8KB的數(shù)據(jù)
- 調(diào)用response.flushBuffer()方法來手動刷新緩沖區(qū)
1.3 設(shè)置響應(yīng)頭信息
可以使用response對象的setHeader()方法來設(shè)置響應(yīng)頭!使用該方法設(shè)置的響應(yīng)頭最終會發(fā)送給客戶端瀏覽器!
response.setHeader(“content-type”, “text/html;charset=utf-8”):設(shè)置content-type響應(yīng)頭,該頭的作用是告訴瀏覽器響應(yīng)內(nèi)容為html類型,編碼為utf-8。而且同時(shí)會設(shè)置response的字符流編碼為utf-8,即response.setCharaceterEncoding(“utf-8”)
response.setHeader(“Refresh”,”5; URL=http://www.itcast.cn“):5秒后自動跳轉(zhuǎn)到傳智主頁
1.4 設(shè)置狀態(tài)碼及其他方法
- response.setContentType(“text/html;charset=utf-8”):等同與調(diào)用response.setHeader(“content-type”, “text/html;charset=utf-8”)
- response.setCharacterEncoding(“utf-8”):設(shè)置字符響應(yīng)流的字符編碼為utf-8
- response.setStatus(200):設(shè)置狀態(tài)碼
- response.sendError(404, “您要查找的資源不存在”):當(dāng)發(fā)送錯(cuò)誤狀態(tài)碼時(shí),Tomcat會跳轉(zhuǎn)到固定的錯(cuò)誤頁面去,但可以顯示錯(cuò)誤信息
1.5 重定向
1.5.1 什么是重定向
當(dāng)你訪問http://www.sun.com時(shí),你會發(fā)現(xiàn)瀏覽器地址欄中的URL會變成http://www.oracle.com/us/sun/index.htm,這就是重定向了。
重定向是服務(wù)器通知瀏覽器去訪問另一個(gè)地址,即再發(fā)出另一個(gè)請求。
1.5.2 完成重定向
響應(yīng)碼為200表示響應(yīng)成功,而響應(yīng)碼為302表示重定向。所以完成重定向的第一步就是設(shè)置響應(yīng)碼為302
因?yàn)橹囟ㄏ蚴峭ㄖ獮g覽器再第二個(gè)請求,所以瀏覽器需要知道第二個(gè)請求的URL,所以完成重定向的第二步是設(shè)置Location頭,指定第二個(gè)請求的URL地址
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {response.setStatus(302);response.setHeader("Location", "http://www.itcast.cn");} }上面代碼的作用是:當(dāng)訪問AServlet后,會通知瀏覽器重定向到傳智主頁。客戶端瀏覽器解析到響應(yīng)碼為302后,就知道服務(wù)器讓它重定向,所以它會馬上獲取響應(yīng)頭Location,然發(fā)出第二個(gè)請求
1.5.3 便捷的重定向方式
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {response.sendRedirect("http://www.itcast.cn");} }response.sendRedirect()方法會設(shè)置響應(yīng)頭為302,以設(shè)置Location響應(yīng)頭
如果要重定向的URL是在同一個(gè)服務(wù)器內(nèi),那么可以使用相對路徑,例如
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {response.sendRedirect("/hello/BServlet");} }重定向的URL地址為:http://localhost:8080/hello/BServlet。
1.5.4 重定向小結(jié)
- 重定向是兩次請求
- 重定向的URL可以是其他應(yīng)用,不局限于當(dāng)前應(yīng)用
- 重定向的響應(yīng)頭為302,并且必須要有Location響應(yīng)頭
- 重定向就不要再使用response.getWriter()或response.getOutputStream()輸出數(shù)據(jù),不然可能會出現(xiàn)異常
2. HttpServletRequest
2.1 Request概述
Request是Servlet.service()方法的一個(gè)參數(shù),類型為javax.servlet.http.HttpServletRequest。在客戶端發(fā)出每個(gè)請求時(shí),服務(wù)器都會創(chuàng)建一個(gè)request對象,并把請求數(shù)據(jù)封裝到request中,然后在調(diào)用Servlet.service()方法時(shí)傳遞給service()方法,這說明在service()方法中可以通過request對象來獲取請求數(shù)據(jù)
Request的功能可以分為以下幾種:
- 封裝了請求頭數(shù)據(jù)
- 封裝了請求正文數(shù)據(jù),如果是GET請求,那么就沒有正文
- request是一個(gè)域?qū)ο?#xff0c;可以把它當(dāng)成Map來添加獲取數(shù)據(jù)
- request提供了請求轉(zhuǎn)發(fā)和請求包含功能
2.2 request域方法
request是域?qū)ο?#xff01;在JavaWeb中一共四個(gè)域?qū)ο?#xff0c;其中ServletContext就是域?qū)ο?#xff0c;它在整個(gè)應(yīng)用中只創(chuàng)建一個(gè)ServletContext對象。request其中一個(gè),request可以在一個(gè)請求中共享數(shù)據(jù)。
一個(gè)請求會創(chuàng)建一個(gè)request對象,如果在一個(gè)請求中經(jīng)歷了多個(gè)Servlet,那么多個(gè)Servlet就可以使用request來共享數(shù)據(jù)。現(xiàn)在我們還不知道如何在一個(gè)請求中經(jīng)歷之幾個(gè)Servlet,后面在學(xué)習(xí)請求轉(zhuǎn)發(fā)和請求包含后就知道了。
下面是request的域方法:
- void setAttribute(String name, Object value)
用來存儲一個(gè)對象,也可以稱之為存儲一個(gè)域?qū)傩?#xff0c;例如:servletContext.setAttribute(“xxx”, “XXX”),在request中保存了一個(gè)域?qū)傩?#xff0c;域?qū)傩悦Q為xxx,域?qū)傩缘闹禐閄XX。請注意,如果多次調(diào)用該方法,并且使用相同的name,那么會覆蓋上一次的值,這一特性與Map相同
- Object getAttribute(String name)
用來獲取request中的數(shù)據(jù),當(dāng)前在獲取之前需要先去存儲才行,例如:String value = (String)request.getAttribute(“xxx”);,獲取名為xxx的域?qū)傩?/p>
- void removeAttribute(String name)
用來移除request中的域?qū)傩?#xff0c;如果參數(shù)name指定的域?qū)傩圆淮嬖?#xff0c;那么本方法什么都不做
- Enumeration getAttributeNames():獲取所有域?qū)傩缘拿Q
2.3 request獲取請求頭數(shù)據(jù)
Request與請求頭相關(guān)的方法有:
| String | getHeader(String name) | 獲取指定名稱的請求頭 |
| Enumeration | getHeaderNames() | 獲取所有請求頭名稱 |
| int | getIntHeader(String name) | 獲取值為int類型的請求頭 |
2.4 request獲取請求相關(guān)的其它方法
request中還提供了與請求相關(guān)的其他方法,有些方法是為了我們更加便捷的方法請求頭數(shù)據(jù)而設(shè)計(jì),有些是與請求URL相關(guān)的方法。
| int | getContentLength() | 獲取請求體的字節(jié)數(shù),GET請求沒有請求體,沒有請求體返回-1 |
| String | getContentType() | 獲取請求類型,如果請求是GET,那么這個(gè)方法返回null;如果是POST請求,那么默認(rèn)為application/x-www-form-urlencoded,表示請求體內(nèi)容使用了URL編碼 |
| String | getMethod() | 返回請求方法,例如:GET |
| Locale | getLocale() | 返回當(dāng)前客戶端瀏覽器的Locale。java.util.Locale表示國家和言語,這個(gè)東西在國際化中很有用 |
| String | getCharacterEncoding() | 獲取請求編碼,如果沒有setCharacterEncoding(),那么返回null,表示使用ISO-8859-1編碼 |
| void | setCharacterEncoding(String code) | 設(shè)置請求編碼,只對請求體有效!注意,對于GET而言,沒有請求體!!!所以此方法只能對POST請求中的參數(shù)有效! |
| String | getContextPath() | 返回上下文路徑,例如:/hello |
| String | getQueryString() | 返回請求URL中的參數(shù),例如:name=zhangSan |
| String | getRequestURI() | 返回請求URI路徑,例如:/hello/oneServlet |
| StringBuffer | getRequestURL() | 返回請求URL路徑,例如:http://localhost/hello/oneServlet,即返回除了參數(shù)以外的路徑信息 |
| String | getServletPath() | 返回Servlet路徑,例如:/oneServlet |
| String | getRemoteAddr() | 返回當(dāng)前客戶端的IP地址 |
| String | getRemoteHost() | 返回當(dāng)前客戶端的主機(jī)名,但這個(gè)方法的實(shí)現(xiàn)還是獲取IP地址 |
| String | getScheme() | 返回請求協(xié)議,例如:http |
| String | getServerName() | 返回主機(jī)名,例如:localhost |
| int | getServerPort() | 返回服務(wù)器端口號,例如:8080 |
2.4.1 案例:request.getRemoteAddr()封IP
可以使用request.getRemoteAddr()方法獲取客戶端的IP地址,然后判斷IP是否為禁用IP。
String ip = request.getRemoteAddr(); System.out.println(ip); if(ip.equals("127.0.0.1")) {response. getWriter().print("您的IP已被禁止!"); } else {response.getWriter().print("Hello!"); }2.5 request獲取請求參數(shù)
最為常見的客戶端傳遞參數(shù)方式有兩種:
- 瀏覽器地址欄直接輸入:一定是GET請求
- 超鏈接:一定是GET請求
- 表單:可以是GET,也可以是POST,這取決與<form>的method屬性值
GET請求和POST請求的區(qū)別:
- GET請求:
- 請求參數(shù)會在瀏覽器的地址欄中顯示,所以不安全
- 請求參數(shù)長度限制長度在1K之內(nèi)
- GET請求沒有請求體,無法通過request.setCharacterEncoding()來設(shè)置參數(shù)的編碼
- POST請求:
- 請求參數(shù)不會顯示瀏覽器的地址欄,相對安全
- 請求參數(shù)長度沒有限制
下面是使用request獲取請求參數(shù)的API:
- String getParameter(String name):通過指定名稱獲取參數(shù)值;
- String[] getParameterValues(String name):當(dāng)多個(gè)參數(shù)名稱相同時(shí),可以使用方法來獲取;
?
public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {String[] names = request.getParameterValues("name");System.out.println(Arrays.toString(names));}- Enumeration getParameterNames():獲取所有參數(shù)的名字;
- Map getParameterMap():獲取所有參數(shù)封裝到Map中,其中key為參數(shù)名,value為參數(shù)值,因?yàn)橐粋€(gè)參數(shù)名稱可能有多個(gè)值,所以參數(shù)值是String[],而不是String。
2.6 請求轉(zhuǎn)發(fā)和請求包含
無論是請求轉(zhuǎn)發(fā)還是請求包含,都表示由多個(gè)Servlet共同來處理一個(gè)請求。例如Servlet1來處理請求,然后Servlet1又轉(zhuǎn)發(fā)給Servlet2來繼續(xù)處理這個(gè)請求。
2.6.1 請求轉(zhuǎn)發(fā)
在AServlet中,把請求轉(zhuǎn)發(fā)到BServlet:
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {System.out.println("AServlet");RequestDispatcher rd = request.getRequestDispatcher("/BServlet");rd.forward(request, response);} } public class BServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {System.out.println("BServlet");} } Aservlet BServlet2.6.2 請求包含
在AServlet中,把請求包含到BServlet:
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {System.out.println("AServlet");RequestDispatcher rd = request.getRequestDispatcher("/BServlet");rd.include(request, response);} } public class BServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {System.out.println("BServlet");} } Aservlet BServlet2.6.3 請求轉(zhuǎn)發(fā)與請求包含比較
- 如果在AServlet中請求轉(zhuǎn)發(fā)到BServlet,那么在AServlet中就不允許再輸出響應(yīng)體,即不能再使用response.getWriter()和response.getOutputStream()向客戶端輸出,這一工作應(yīng)該由BServlet來完成;如果是使用請求包含,那么沒有這個(gè)限制
- 請求轉(zhuǎn)發(fā)雖然不能輸出響應(yīng)體,但還是可以設(shè)置響應(yīng)頭的,例如:response.setContentType(”text/html;charset=utf-8”);
- 請求包含大多是應(yīng)用在JSP頁面中,完成多頁面的合并
請求轉(zhuǎn)發(fā)大多是應(yīng)用在Servlet中,轉(zhuǎn)發(fā)目標(biāo)大多是JSP頁面
2.6.4 請求轉(zhuǎn)發(fā)與重定向比較
- 請求轉(zhuǎn)發(fā)是一個(gè)請求,而重定向是兩個(gè)請求
- 請求轉(zhuǎn)發(fā)后瀏覽器地址欄不會有變化,而重定向會有變化,因?yàn)橹囟ㄏ蚴莾蓚€(gè)請求
- 請求轉(zhuǎn)發(fā)的目標(biāo)只能是本應(yīng)用中的資源,重定向的目標(biāo)可以是其他應(yīng)用
- 請求轉(zhuǎn)發(fā)對AServlet和BServlet的請求方法是相同的,即要么都是GET,要么都是POST,因?yàn)檎埱筠D(zhuǎn)發(fā)是一個(gè)請求
- 重定向的第二個(gè)請求一定是GET
3. 路徑
3.1 與路徑相關(guān)的操作
- 超鏈接
- 表單
- 轉(zhuǎn)發(fā)
- 包含
- 重定向
- <url-pattern>
- ServletContext獲取資源
- Class獲取資源
- ClassLoader獲取資源
3.2 客戶端路徑
超鏈接、表單、重定向都是客戶端路徑,客戶端路徑可以分為三種方式:
- 絕對路徑
- 以“/”開頭的相對路徑
- 不以“/”開頭的相對路徑
例如:http://localhost:8080/hello1/pages/a.html中的超鏈接和表單如下:
<!--絕對路徑--> <a href="http://localhost:8080/hello2/index.html">鏈接1</a><!--客戶端路徑--> <a href="/hello3/pages/index.html">鏈接2</a><!--相對路徑--> <a href="index.html">鏈接3</a> <!--絕對路徑--> <form action="http://localhost:8080/hello2/index.html"><input type="submit" value="表單1"/> </form><!--客戶端路徑--> <form action="/hello2/index.html"><input type="submit" value="表單2"/> </form><!--相對路徑--> <form action="index.html"><input type="submit" value="表單3"/> </form>- 鏈接1和表單1:沒什么可說的,它使用絕對路徑
- 鏈接2和表單2:以“/”開頭,相對主機(jī),與當(dāng)前a.html的主機(jī)相同,即最終訪問的頁面為http://localhost:8080/hello2/index.html
- 鏈接3和表單3:不以“/”開頭,相對當(dāng)前頁面的路徑,即a.html所有路徑,即最終訪問的路徑為:http://localhost:8080/hello1/pages/index.html
重定向1
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {response.sendRedirect("/hello/index.html");} }假設(shè)訪問AServlet的路徑為:http://localhost:8080/hello/servlet/AServlet
因?yàn)槁窂揭浴?”開頭,所以相對當(dāng)前主機(jī),即http://localhost:8080/hello/index.html
重定向2
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {response.sendRedirect("index.html");} }假設(shè)訪問AServlet的路徑為:http://localhost:8080/hello/servlet/AServlet
因?yàn)槁窂讲灰浴?”開頭,所以相對當(dāng)前路徑,即http://localhost:8080/hello/servlet/index.html
3.3 建議使用“/”
強(qiáng)烈建議使用“/”開頭的路徑,這說明在頁面中的超鏈接和表單都要以“/”開頭,后面是當(dāng)前應(yīng)用的名稱,再是訪問路徑:
<form action="/hello/servlet/AServlet"></form> <a href="/hello/b.html">鏈接</a>其中/hello是當(dāng)前應(yīng)用名稱,這也說明如果將來修改了應(yīng)用名稱,那么頁面中的所有路徑也要修改,這一點(diǎn)確實(shí)是個(gè)問題。這一問題的處理方案會在學(xué)習(xí)了JSP之后講解!
在Servlet中的重定向也建議使用“/”開頭。同理,也要給出應(yīng)用的名稱!例如:
response.sendRedirect("/hello/BServlet")其中/hello是當(dāng)前應(yīng)用名,如果將來修改了應(yīng)用名稱,那么也要修改所有重定向的路徑,這一問題的處理方案是使用request.getContextPath()來獲取應(yīng)用名稱
response.sendRedirect(request.getContextPath() + "/BServlet")3.4 服務(wù)器端路徑
服務(wù)器端路徑必須是相對路徑,不能是絕對路徑。但相對路徑有兩種形式:
- 以“/”開頭
- 不以“/”開頭
其中請求轉(zhuǎn)發(fā)、請求包含都是服務(wù)器端路徑,服務(wù)器端路徑與客戶端路徑的區(qū)別是:
- 客戶端路徑以“/”開頭:相對當(dāng)前主機(jī)
- 服務(wù)器端路徑以“/”開頭:相對當(dāng)前應(yīng)用
轉(zhuǎn)發(fā)1:
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {request.getRequestDispatcher("/BServlet").forward(request, response);} }假設(shè)訪問AServlet的路徑為:http://localhost:8080/hello/servlet/AServlet
因?yàn)槁窂揭浴?”開頭,所以相對當(dāng)前應(yīng)用,即http://localhost:8080/hello/BServlet。
轉(zhuǎn)發(fā)2:
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {request.getRequestDispatcher("BServlet").forward(request, response);} }假設(shè)訪問AServlet的路徑為:http://localhost:8080/hello/servlet/AServlet
因?yàn)槁窂讲灰浴?”開頭,所以相對當(dāng)前應(yīng)用,即http://localhost:8080/hello/servlet/BServlet。
3.5 <url-pattern>路徑
<url-pattern>必須使用“/”開頭,并且相對的是當(dāng)前應(yīng)用。
3.6 ServletContext獲取資源
必須是相對路徑,可以“/”開頭,也可以不使用“/”開頭,但無論是否使用“/”開頭都是相對當(dāng)前應(yīng)用路徑
例如在AServlet中獲取資源,AServlet的路徑路徑為:http://localhost:8080/hello/servlet/AServlet:
public class AServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {String path1 = this.getServletContext().getRealPath("a.txt");String path2 = this.getServletContext().getRealPath("/a.txt");System.out.println(path1);System.out.println(path2);} }path1和path2是相同的結(jié)果:http://localhost:8080/hello/a.txt
3.7 Class獲取資源
Class獲取資源也必須是相對路徑,可以“/”開頭,也可以不使用“/”開頭。
package cn.itcast;import java.io.InputStream;public class Demo {public void fun1() {InputStream in = Demo.class.getResourceAsStream("/a.txt");}public void fun2() {InputStream in = Demo.class.getResourceAsStream("a.txt");} }其中fun1()方法獲取資源時(shí)以“/”開頭,那么相對的是當(dāng)前類路徑,即/hello/WEB-INF/classes/a.txt文件
其中fun2()方法獲取資源時(shí)沒有以“/”開頭,那么相對當(dāng)前Demo.class所在路徑,因?yàn)镈emo類在cn.itcast包下,所以資源路徑為:/hello/WEB-INF/classes/cn/itcast/a.txt
3.8 ClassLoader獲取資源
ClassLoader獲取資源也必須是相對路徑,可以“/”開頭,也可以不使用“/”開頭。但無論是否以“/”開頭,資源都是相對當(dāng)前類路徑。
public class Demo {public void fun1() {InputStream in = Demo.class.getClassLoader().getResourceAsStream("/a.txt");}public void fun2() {InputStream in = Demo.class.getClassLoader().getResourceAsStream("a.txt");} }fun1()和fun2()方法的資源都是相對類路徑,即classes目錄,即/hello/WEB-INF/classes/a.txt
4. 編碼
4.1 請求編碼
4.1.1 直接在地址欄中給出中文
請求數(shù)據(jù)是由客戶端瀏覽器發(fā)送服務(wù)器的,請求數(shù)據(jù)的編碼是由瀏覽器決定的。例如在瀏覽器地址欄中給出:http://localhost:8080/hello/AServlet?name=傳智,那么其中“傳智”是什么編碼的呢?不同瀏覽器使用不同的編碼,所以這是不確定的!
- IE:使用GB2312
- FireFox:使用GB2312
- Chrome:使用UTF-8
通常沒有哪個(gè)應(yīng)用要求用戶在瀏覽器地址欄中輸入請求數(shù)據(jù)的,所以大家只需了解一下即可。
4.1.2 在頁面中發(fā)出請求
通常向服務(wù)器發(fā)送請求數(shù)據(jù)都需要先請求一個(gè)頁面,然后用戶在頁面中輸入數(shù)據(jù)。頁面中有超鏈接和表單,通過超鏈接和表單就可以向服務(wù)器發(fā)送數(shù)據(jù)了。
因?yàn)轫撁媸欠?wù)器發(fā)送到客戶端瀏覽器的,所以這個(gè)頁面本身的編碼由服務(wù)器決定。而用戶在頁面中輸入的數(shù)據(jù)也是由頁面本身的編碼決定的。
index.html
當(dāng)用戶在index.html頁面中輸入數(shù)據(jù)時(shí),都是UTF-8列表的。因?yàn)檫@個(gè)頁面本身就是UTF-8編碼的!
頁面的編譯就是頁面中輸入數(shù)據(jù)的編碼。
4.1.3 GET請求解讀編碼
當(dāng)客戶端通過GET請求發(fā)送數(shù)據(jù)給服務(wù)器時(shí),使用request.getParameter()獲取的數(shù)據(jù)是被服務(wù)器誤認(rèn)為ISO-8859-1編碼的,也就是說客戶端發(fā)送過來的數(shù)據(jù)無論是UTF-8還是GBK,服務(wù)器都認(rèn)為是ISO-8859-1,這就說明我們需要在使用request.getParameter()獲取數(shù)據(jù)后,再轉(zhuǎn)發(fā)成正確的編碼。
例如客戶端以UTF-8發(fā)送的數(shù)據(jù),使用如下轉(zhuǎn)碼方式:
String name = request.getParameter(“name”); name = new String(name.getBytes(“iso-8859-1”), “utf-8”);4.1.4 POST請求解讀編碼
當(dāng)客戶端通過POST請求發(fā)送數(shù)據(jù)給服務(wù)器時(shí),可以在使用request.getParameter()獲取請求參數(shù)之前先通過request.setCharacterEncoding()來指定編碼,然后再使用reuqest.getParameter()方法來獲取請求參數(shù),那么就是用指定的編碼來讀取了。
也就是說,如果是POST請求,服務(wù)器可以指定編碼!但如果沒有指定編碼,那么默認(rèn)還是使用ISO-8859-1來解讀。
request.setCharacterEncoding(“utf-8”); String name = request.getParameter(“name”);4.2 響應(yīng)編碼
響應(yīng):服務(wù)器發(fā)送給客戶端數(shù)據(jù)!響應(yīng)是由response對象來完成,如果響應(yīng)的數(shù)據(jù)不是字符數(shù)據(jù),那么就無需去考慮編碼問題。當(dāng)然,如果響應(yīng)的數(shù)據(jù)是字符數(shù)據(jù),那么就一定要考慮編碼的問題了。
response.getWriter().print(“傳智”);上面代碼因?yàn)闆]有設(shè)置repsonse.getWriter()字符流的編碼,所以服務(wù)器使用默認(rèn)的編碼(ISO-8859-1)來處理,因?yàn)镮SO-8859-1不支持中文,所以一定會出現(xiàn)編碼的。
所以在使用response.getWriter()發(fā)送數(shù)據(jù)之前,一定要設(shè)置response.getWriter()的編碼,這需要使用
上面代碼因?yàn)樵谑褂胷esponse.getWriter()輸出之前已經(jīng)設(shè)置了編碼,所以輸出的數(shù)據(jù)為utf-8編碼。但是,因?yàn)闆]有告訴瀏覽器使用什么編碼來讀取響應(yīng)數(shù)據(jù),所以很可能瀏覽器會出現(xiàn)錯(cuò)誤的解讀,那么還是會出現(xiàn)亂碼的。當(dāng)然,通常瀏覽器都支持來設(shè)置當(dāng)前頁面的編碼,如果用戶在看到編碼時(shí),去設(shè)置瀏覽器的編碼,如果設(shè)置的正確那么亂碼就會消失。但是我們不能讓用戶總?cè)プ约涸O(shè)置編碼,而且應(yīng)該直接通知瀏覽器,服務(wù)器發(fā)送過來的數(shù)據(jù)是什么編碼,這樣瀏覽器就直接使用服務(wù)器告訴他的編碼來解讀!這需要使用content-type響應(yīng)頭。
response.setContentType(“text/html;charset=utf-8”); response.getWriter().print(“傳智”);上面代碼使用setContentType()方法設(shè)置了響應(yīng)頭content-type編碼為utf-8,這不只是在響應(yīng)中添加了響應(yīng)頭,還等于調(diào)用了一次response.setCharacterEncoding(“utf-8”),也就是說,通過我們只需要調(diào)用一次response.setContentType(“text/html;charset=utf-8”)即可,而無需再去調(diào)用response.setCharacterEncoding(“utf-8”)了。
在靜態(tài)頁面中,使用<meta>來設(shè)置content-type響應(yīng)頭,例如:
<meta http-equiv=”content-type” content=”text/html; charset=UTF-8”>
4.3 URL編碼
通過頁面?zhèn)鬏敂?shù)據(jù)給服務(wù)器時(shí),如果包含了一些特殊字符是無法發(fā)送的。這時(shí)就需要先把要發(fā)送的數(shù)據(jù)轉(zhuǎn)換成URL編碼格式,再發(fā)送給服務(wù)器。
其實(shí)需要我們自己動手給數(shù)據(jù)轉(zhuǎn)換成URL編碼的只有GET超鏈接,因?yàn)楸韱伟l(fā)送數(shù)據(jù)會默認(rèn)使用URL編碼,也就是說,不用我們自己來編碼。
例如:“傳智”這兩個(gè)字通過URL編碼后得到的是:“%E4%BC%A0%E6%99%BA”。URL編碼是先需要把“傳智”轉(zhuǎn)換成字節(jié),例如我們現(xiàn)在使用UTF-8把“傳智”轉(zhuǎn)換成字符,得到的結(jié)果是:“[-28, -68, -96, -26, -103, -70]”,然后再把所有負(fù)數(shù)加上256,得到[228, 188, 160, 230, 153, 186],再把每個(gè)int值轉(zhuǎn)換成16進(jìn)制,得到[E4, BC, A0, E6, 99, BA],最后再每個(gè)16進(jìn)制的整數(shù)前面加上“%”。
通過URL編碼,把“傳智”轉(zhuǎn)換成了“%E4%BC%A0%E6%99%BA”,然后發(fā)送給服務(wù)器!服務(wù)器會自動識別出數(shù)據(jù)是使用URL編碼過的,然后會自動把數(shù)據(jù)轉(zhuǎn)換回來。
當(dāng)然,在頁面中我們不需要自己去通過上面的過程把“傳智”轉(zhuǎn)換成“%E4%BC%A0%E6%99%BA”,而是使用Javascript來完成即可。當(dāng)后面我們學(xué)習(xí)了JSP后,就不用再使用Javascript了。
<script type="text/javascript">function _go() {location = "/day05_2/AServlet?name=" + encodeURIComponent("傳智+播客");}</script> <a href="javascript:_go();">鏈接</a>因?yàn)閁RL默認(rèn)只支持ISO-8859-1,這說明在URL中出現(xiàn)中文和一些特殊字符可能無法發(fā)送到服務(wù)器。所以我們需要對包含中文或特殊字符的URL進(jìn)行URL編碼。
服務(wù)器會自動識別數(shù)據(jù)是否使用了URL編碼,如果使用了服務(wù)器會自動把數(shù)據(jù)解碼,無需我們自己動手解碼。
總結(jié)
以上是生活随笔為你收集整理的JavaWeb:HttpServletResponse和HttpServletRequest的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: JavaWeb:用JDBC操作数据库
- 下一篇: JavaWeb:JSP