Servlet初识了解和应用
Servlet
Servlet 是什么?
Java Servlet 是運行在 Web 服務器或應用服務器上的程序,它是作為來自 Web 瀏覽器或其他 HTTP 客戶端的請求和 HTTP 服務器上的數據庫或應用程序之間的中間層。
使用 Servlet,您可以收集來自網頁表單的用戶輸入,呈現來自數據庫或者其他源的記錄,還可以動態創建網頁。
Java Servlet 通常情況下與使用 CGI(Common Gateway Interface,公共網關接口)實現的程序可以達到異曲同工的效果。但是相比于 CGI,Servlet 有以下幾點優勢:
- 性能明顯更好。
- Servlet 在 Web 服務器的地址空間內執行。這樣它就沒有必要再創建一個單獨的進程來處理每個客戶端請求。
- Servlet 是獨立于平臺的,因為它們是用 Java 編寫的。
- 服務器上的 Java 安全管理器執行了一系列限制,以保護服務器計算機上的資源。因此,Servlet 是可信的。
- Java 類庫的全部功能對 Servlet 來說都是可用的。它可以通過 sockets 和 RMI 機制與 applets、數據庫或其他軟件進行交互。
Servlet 架構
下圖顯示了 Servlet 在 Web 應用程序中的位置。
Servlet 任務
Servlet 執行以下主要任務:
- 讀取客戶端(瀏覽器)發送的顯式的數據。這包括網頁上的 HTML 表單,或者也可以是來自 applet 或自定義的 HTTP 客戶端程序的表單。
- 讀取客戶端(瀏覽器)發送的隱式的 HTTP 請求數據。這包括 cookies、媒體類型和瀏覽器能理解的壓縮格式等等。
- 處理數據并生成結果。這個過程可能需要訪問數據庫,執行 RMI 或 CORBA 調用,調用 Web 服務,或者直接計算得出對應的響應。
- 發送顯式的數據(即文檔)到客戶端(瀏覽器)。該文檔的格式可以是多種多樣的,包括文本文件(HTML 或 XML)、二進制文件(GIF 圖像)、Excel 等。
- 發送隱式的 HTTP 響應到客戶端(瀏覽器)。這包括告訴瀏覽器或其他客戶端被返回的文檔類型(例如 HTML),設置 cookies 和緩存參數,以及其他類似的任務。
Servlet 包
Java Servlet 是運行在帶有支持 Java Servlet 規范的解釋器的 web 服務器上的 Java 類。
Servlet 可以使用 javax.servlet 和 javax.servlet.http 包創建,它是 Java 企業版的標準組成部分,Java 企業版是支持大型開發項目的 Java 類庫的擴展版本。
這些類實現 Java Servlet 和 JSP 規范。在寫本教程的時候,二者相應的版本分別是 Java Servlet 2.5 和 JSP 2.1。
Java Servlet 就像任何其他的 Java 類一樣已經被創建和編譯。在您安裝 Servlet 包并把它們添加到您的計算機上的 Classpath 類路徑中之后,您就可以通過 JDK 的 Java 編譯器或任何其他編譯器來編譯 Servlet。
##Servlet生命周期
Servlet 生命周期可被定義為從創建直到毀滅的整個過程。以下是 Servlet 遵循的過程:
- Servlet 初始化后調用 init () 方法。
- Servlet 調用 service() 方法來處理客戶端的請求。
- Servlet 銷毀前調用 destroy() 方法。
- 最后,Servlet 是由 JVM 的垃圾回收器進行垃圾回收的。
現在讓我們詳細討論生命周期的方法。
init() 方法
init 方法被設計成只調用一次。它在第一次創建 Servlet 時被調用,在后續每次用戶請求時不再調用。因此,它是用于一次性初始化,就像 Applet 的 init 方法一樣。
Servlet 創建于用戶第一次調用對應于該 Servlet 的 URL 時,但是您也可以指定 Servlet 在服務器第一次啟動時被加載。
當用戶調用一個 Servlet 時,就會創建一個 Servlet 實例,每一個用戶請求都會產生一個新的線程,適當的時候移交給 doGet 或 doPost 方法。init() 方法簡單地創建或加載一些數據,這些數據將被用于 Servlet 的整個生命周期。
init 方法的定義如下:
@Overridepublic void init() throws ServletException {System.out.println("servlet初始化");}第一次訪問servlet:
再次訪問:
說明:servlet的init()只在訪問servlet第一次被初始化,只要服務器沒有關閉,則只會初始化一次。
如果在web.xml配置了 <load-on-startup>1</load-on-startup> 則該servlet在服務器啟動時就被加載。
service() 方法
service() 方法是執行實際任務的主要方法。Servlet 容器(即 Web 服務器)調用 service() 方法來處理來自客戶端(瀏覽器)的請求,并把格式化的響應寫回給客戶端。
每次服務器接收到一個 Servlet 請求時,服務器會產生一個新的線程并調用服務。service() 方法檢查 HTTP 請求類型(GET、POST、PUT、DELETE 等),并在適當的時候調用 doGet、doPost、doPut,doDelete 等方法。
下面是該方法的特征:
@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {resp.getWriter().write("servlet life");System.out.println("servlet life");}service() 方法由容器調用,service 方法在適當的時候調用 doGet、doPost、doPut、doDelete 等方法。所以,您不用對 service() 方法做任何動作,您只需要根據來自客戶端的請求類型來重寫 doGet() 或 doPost() 即可。
doGet() 和 doPost() 方法是每次服務請求中最常用的方法。下面是這兩種方法的特征。
doGet() 方法
GET 請求來自于一個 URL 的正常請求,或者來自于一個未指定 METHOD 的 HTML 表單,它由 doGet() 方法處理。
public void doGet(HttpServletRequest request,HttpServletResponse response)throws ServletException, IOException {// Servlet 代碼 }簡單登陸:
<form action="servletmethod" method="get">用戶名:<input type="text" name="uname" ><br>密碼:<input type="text" name="password"><br><input type="submit" value="登陸"></form>點擊登陸后搜索框:
即:對用戶是可見的,不安全,當同時有doGet方法和service方法時,只調用service方法。
doPost() 方法
POST 請求來自于一個特別指定了 METHOD 為 POST 的 HTML 表單,它由 doPost() 方法處理。
public void doPost(HttpServletRequest request,HttpServletResponse response)throws ServletException, IOException {// Servlet 代碼 } <form action="servletmethod" method="post">用戶名:<input type="text" name="uname" ><br>密碼:<input type="text" name="password"><br><input type="submit" value="登陸"></form>使用post請求:
對用戶不可現,更加安全。
destroy() 方法
destroy() 方法只會被調用一次,在 Servlet 生命周期結束時被調用。destroy() 方法可以讓您的 Servlet 關閉數據庫連接、停止后臺線程、把 Cookie 列表或點擊計數器寫入到磁盤,并執行其他類似的清理活動。
在調用 destroy() 方法之后,servlet 對象被標記為垃圾回收。destroy 方法定義如下所示:
@Overridepublic void destroy() {System.out.println("servlet 銷毀");}當停止tomcat服務器時:
說明servlet在服務器關閉時被銷毀。
架構圖
下圖顯示了一個典型的 Servlet 生命周期方案。
- 第一個到達服務器的 HTTP 請求被委派到 Servlet 容器。
- Servlet 容器在調用 service() 方法之前加載 Servlet。
- 然后 Servlet 容器處理由多個線程產生的多個請求,每個線程執行一個單一的 Servlet 實例的 service() 方法。
注意
如果在復寫的service方法中調用了父類的service方法(super.service(arg0,arg1)),則service方法處理完后,會再次根據請求方式響應的doGet和doPost方法執行。所以,一般情況下,我們是不在復寫的service中調用父類打的service方法的,避免出現405錯誤。
Servlet常見錯誤
404錯誤:資源未找到
? 原因1:在請求地址中的servlet的別名書寫錯誤。
? 原因2:虛擬項目名拼寫錯誤。
500錯誤:內部服務器錯誤
? 錯誤1:java.lang.ClassNotFoundException: com.youxin.servlet.ServletMethod
? 解決:
? 在web.xml中校驗servlet類的全限路徑是否拼寫錯誤
? 錯誤2:因為service方法體的代碼執行導致錯誤
? 解決:
? 根據錯誤提示對service方法體中的代碼進行錯誤比較
405錯誤:請求方式不支持
? 原因:請求方式和servlet中的方法不匹配造成的
? 解決:盡量使用service方法進行請求處理,并且不要在service方法中調用父類的service方法。
Request和Response對象
概述:
我們在創建Servlet時會覆蓋service()方法,或doGet()/doPost(),這些方法都有兩個參數,一個為代表請求的request和代表響應的response。
service方法中的request的類型是ServletRequest,而doGet/doPost方法的request的類型是HttpServletRequest,HttpServletRequest是ServletRequest的子接口,功能和方法更加強大。
service方法中的response的類型是ServletResponse,而doGet/doPost方法的response的類型是HttpServletResponse,HttpServletResponse是ServletResponse的子接口,功能和方法更加強大。
###Request對象
request獲取請求頭
/* * request對象 * 作用:request對象封存了當前請求的所有請求信息 * 使用: * 獲取請求頭數據 * req.getMethod(); * req.getRequestURL(); * req.getRequestURI(); * req.getScheme(); * 獲取請求行數據 * req.getHeader("Accept-Language"); * 獲取用戶數據 * req.getParameter() * req.getParameterValues("鍵名");//返回同鍵不同值的請求數據(多選),返回的數組 * req.getParameterNames();//返回所有用戶請求數據的枚舉集合 * 注意: * request對象由tomcat服務器創建,并作為實參傳遞給處理請求的servlet的service方法 * */ public class RequestServlet extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//獲取請求頭數據//獲取請求方式String method = req.getMethod();System.out.println(method);//獲取請求URLStringBuffer requestURL = req.getRequestURL();System.out.println(requestURL);//獲取URIString requestURI = req.getRequestURI();System.out.println(requestURI);//獲取協議String scheme = req.getScheme();System.out.println(scheme);} }輸出:
request獲取請求行
//獲取請求行數據//獲取指定的請求行信息String value = req.getHeader("Accept-Language");System.out.println(value);//獲取所有的請求行的鍵的枚舉Enumeration headerNames = req.getHeaderNames();while (headerNames.hasMoreElements()){String header = req.getHeader(headerNames.nextElement().toString());System.out.print("請求頭的鍵:"+headerNames.nextElement()+",");System.out.println("請求頭的值:"+header);}結果:
//不管是get還是post都用req.getParameter()獲取用戶屬性值//獲取用戶信息String uname = req.getParameter("uname");String password = req.getParameter("password");System.out.println("uname:"+uname+",password:"+password);getparameter不能獲取同鍵不同值的對象
如果出現同鍵不同值的情況則需要用 req.getParameterValues() 獲取
如:
愛好:<br><input type="checkbox" name="fav" value="1">唱歌<input type="checkbox" name="fav" value="2">跳舞<input type="checkbox" name="fav" value="3">打游戲獲取:
String[] favs = req.getParameterValues("fav"); if (favs != null){for (String s: favs) {System.out.println(s);} }Response對象:
問題:在使用Requset對象獲取了請求數據并進行處理后,處理的結果如何顯示到瀏覽器中呢?
解決:使用Reponse對象
解釋:服務器在調用指定的servlet進行請求處理的時候,會給servlet的方法傳遞兩個實參request和response。其中requset中封存了請求相關的請求數據,而response則是用來進行相應的一個對象。
/* * * Response * 作用:響應數據到瀏覽器的一個對象 * 使用:設置響應頭 * setHeader()//在響應頭中添加響應信息,但同鍵會覆蓋 * addHeader()//在響應頭中添加信息,但不會覆蓋 * 設置響應狀態:sendError();//自定義響應狀態碼 * 設置響應體 * resp.getWriter().write() * 設置響應編碼格式: * resp.setContentType("text/html;charset=utf-8") * */ public class ResponseServlet extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//獲取請求信息//獲取請求頭//獲取請求行//獲取用戶數據//處理請求//響應處理結果//設置響應頭resp.setHeader("mouse", "leigod");resp.addHeader("key", "leigod");//設置編碼格式 // resp.setCharacterEncoding("utf-8");resp.setContentType("text/html;charset=utf-8");//設置響應狀態碼 // resp.sendError(404,"sorry"); // resp.sendError(405,"this method is not supported");//設置響應實體resp.getWriter().write("<a href='http://www.baidu.com'>百度一下</a>");} }####請求轉發和重定向的區別
1.請求轉發是服務器內部跳轉,地址欄不會發生改變
重定向地址欄會發生改變。
2.請求轉發,只有一次請求,一次響應.
重定向,有兩次請求,兩次響應。
3.請求轉發存在request域,可以共享數據.
重定向不存在request域。
4.請求轉發只能在服務器的內部跳轉,簡單說,只能訪問本站內資源。
重定向可以訪問站外資源,也可以訪問站內資源.
5.請求轉發是由request 發起的 . request.getRequestDispatcher().forward()
重定向是由response 發起的 response.sendRedirect();
6.請求轉發與重定向時路徑寫法不一樣.
重定向要跳轉的路徑是瀏覽器在次發起的,是瀏覽器端路徑:寫法: /工程名/資源
請求轉發是服務器內部跳轉,這時它與瀏覽器無關 寫法:/資源
RequestDispatcher.include()方法用于將RequestDispatcher對象封裝的資源內容作為當前響應內容的一部分包含進來,從而實現可編程的服務器端包含功能
被包含的Servlet程序不能改變響應消息的狀態碼和響應頭,如果它里面存在這樣的語句,這些語句的執行結果將被忽略.include在程序執行上效果類似forward,但是使用forward只有一個程序可以生成響應,include可以由多個程序一同生成響應 ----- 常用來頁面布局
New兩個html頁面分別為 i1.html&i2.html
i1內容1111111111
i2內容22222222222
New一個Servlet get方法下
request.getRequestDispatcher(“/i1.html”).include(request,response);
request.getRequestDispatcher(“/i2.html”).include(request,response);
最后在頁面輸出的內容為111111111122222222222是兩個頁面的所用內容.
總結
以上是生活随笔為你收集整理的Servlet初识了解和应用的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 【深度学习】常见的提高模型泛化能力的方法
- 下一篇: 批量创建文件夹并命名