Jsp技術總結
這節我們總結一下Jsp的相關技術。
1.什麼是JSPJSP即JavaServer Pages,它和servlet技術一樣,都是sun公司定義的一種用於開發動態web資源的技術。該技術的最大特點在於:寫JSP就像寫html,但它相比html而言,html只能為用戶提供靜態數據,而JSP技術允許在頁面中嵌套java代碼,為用戶提供動態數據。
2. JSP原理1) web伺服器是如何調用並執行一個JSP頁面的?伺服器首先將JSP翻譯為一個Servlet,翻譯過後的Servlet可以在 omcat主目錄workCatalinalocalhost工程名orgapachejsp目錄下查看,這是伺服器的工作目錄。打開相應的Servlet可以看到,翻譯過後的servlet繼承了org.apache.jasper.runtime.HttpJspBase,而HttpJspBase繼承了HttpSerrvlet。說到這裡,我們就明白了,其實JSP就是一個Servlet,訪問jsp即訪問一個Servlet。
2) JSP頁面中的html排版標籤是如何被發送到客戶端的以及java代碼伺服器是如何執行的?繼續瀏覽JSP翻譯過後的Servlet,裡面有個service方法_jspService(request, response),在該方法中可以看到,JSP中的所有內容都會翻譯到service方法中,html代碼會通過out輸出,就像學習JSP之前,在Servlet中用out輸出html語句一樣,java部分代碼會原封不動的搬到service方法中。
3) web伺服器在調用JSP時,會給JSP提供一些什麼java對象?web伺服器在將JSP翻譯成Servlet時,會在service方法中提供web開發所有的對象,這樣在JSP頁面中,我們就可以直接使用這些對象了,而不用再去獲得。web伺服器所提供的對象(這些對象我們下面戶詳細分析)有:
final javax.servlet.jsp.PageContext pageContext; //pageContext對象 javax.servlet.http.HttpSession session = null; //session對象 final javax.servlet.ServletContext application; //application對象 final javax.servlet.ServletConfig config; //config對象 javax.servlet.jsp.JspWriter out = null; //out對象 final java.lang.Object page = this; //page對象 javax.servlet.jsp.JspWriter _jspx_out = null; javax.servlet.jsp.PageContext _jspx_page_context = null;
還有service方法的參數request和response。所以Servlet可以做的事,JSP都可以做。但是兩者各有特點,我們繼續往下看。
3. JSP的最佳實踐不管是JSP還是Servlet,雖然都可以用於開發動態web資源,但是由於這2門技術各自的特點,在長期的軟體實踐中,人們逐漸把servlet作為web應用中的控制器組件來使用,而把JSP技術作為數據顯示模板來使用。原因在於,程序的數據通常要美化後再輸出。讓JSP既用java代碼產生動態數據,又做美化會導致頁面難以維護;讓Servlet既產生數據,又在裡面嵌套html代碼美化數據,同樣也會導致程序可讀性差,難以維護;因此最好的辦法就是讓Servlet負責相應請求產生的數據,並把數據通過轉發技術帶給JSP,JSP用來做數據的顯示。
4. JSP語法1) JSP模板元素JSP模板元素即JSP頁面中的HTML內容。JSP模板元素定義了網頁的基本骨架,即定義了頁面的結構和外觀。
2) JSP腳本表達式JSP腳本表達式用於將程序數據輸出到客戶端。
語法:<%= 變數或表達式 %>如:當前時間:<%= new java.util.Date() %> JSP引擎在翻譯腳本表達式時,會將程序數據轉成字元串,然後在相應位置用out.print(...)將數據輸給客戶端。 注意:JSP腳本表達式中的變數或表達式後面不能有分號(;)
3) JSP腳本片段JSP腳本片段用於在JSP頁面中編寫多行java代碼。 語法:<% 多行java代碼 %> 注意:JSP腳本片段只能出現java代碼,不能出現其他模板元素,JSP引擎在翻譯JSP頁面中,會將JSP腳本片段中的Java代碼原封不動的放到Servlet的_jspService方法中。JSP腳本片段中的java代碼必須嚴格遵循java語法。 在一個JSP頁面中可以有多個腳本片段,在兩個或多個腳本片段之間可以嵌入文本、HTML標記和其他JSP元素。不同腳本片段中的代碼可以相互訪問,猶如將所有的代碼放到一個<% %>中一樣。但是所有的腳本片段組合在一起必須是一個完整的java代碼。
4) JSP聲明JSP聲明中的java代碼會被翻譯到_jspService方法的外面。(面試題) 語法:<%! java代碼 %> 所以,JSP聲明可用於定義JSP頁面轉換成Servlet程序的靜態代碼塊、成員變數和方法。多個靜態代碼塊、變數和方法可以定義在一個JSP聲明中,也可以分別單獨定義在多個JSP聲明中。 JSP隱式對象的作用範圍僅限於Servlet的_jspService方法中,而JSP聲明的代碼會被翻譯到該方法的外面,所以在JSP聲明中不能使用這些隱式對象。
5) JSP注釋語法:<%- 注釋信息 -%> JSP引擎在將JSP頁面翻譯成Servlet時,忽略JSP頁面中被注釋的內容。 如果在JSP中使用<!-- 注釋內容-->時,注釋內容會打給瀏覽器,瀏覽器認識這個注釋,所以不會顯示給用戶。而JSP注釋將注釋內容不打給瀏覽器。
6) JSP指令JSP指令是為JSP引擎而設計的,它們並不直接產生任何可見輸出,而只是告訴引擎如何處理JSP頁面中其餘的部分。 語法:<%@ 指令 屬性名="值" %> 如:<%@ page contentType="text/html;charset=UTF-8" %> <%@ page import="java.util.Date" %> 如果一個指令有多個屬性,這多個屬性可以寫在一個指令中,中間用空格隔開。即上面兩條指令等價表示如下:
<%@ page contentType="text/html;charset=UTF-8" import="java.util.Date" %>
在JSP2.0規範中共定義了三個指令: 1)page指令:
page指令用於定義JSP頁面的各種屬性,無論page指令出現在JSP頁面中的什麼地方,它的作用都是整個JSP頁面。為了保持程序的可讀性,page指令最好放在整個JSP頁面的起始位置。 語法:
<%@ page [import="{package.class | package.*},..."] [session="true | false"] [errorPage=""relative_url] [isErrorPage="true | false"] [contentType="mimeType[;charset=characterSet]" | "text/html;charset=ISO-8859-1] [pageEncoding="characterSet | ISO-8859-1"] [isELIgnored="true | false"] %>
JSP引擎會自動導入如下包:java.lang.* / java.servlet.* / javax.servlet.jsp.* / javax.servlet.http.* JSP導入多個包的時候,可以分開寫,也可以放在一起寫,放在一起的時候,使用逗號分隔:
<%@ page import="java.util.Date,java.sql.*,java.io.*"%>
session屬性設置為true時,翻譯後的servlet中會自動創建session對象,false則不創建。 errorPage屬性用來設置錯誤相應頁面。它的值必須使用相對路徑,如果以"/"開頭,表示相對於當前web應用程序的根目錄(注意不是站點根目錄),否則,表示相對於當前頁面。也可以在web.xml文件中使用<error-page>元素為整個web應用程序設置錯誤處理頁面,其中<exception-type>子元素指定異常類的完全限定名,<location>元素指定以"/"開頭的錯誤處理頁面路徑。如果設置了某個JSP頁面的errorPage屬性,那麼在web.xml文件中設置的錯誤處理將不對該頁面起作用。 isErrorPage為true時,表示該頁面是處理錯誤頁面,JSP引擎在翻譯成servlet時,會定義一個exception對象,從而就可以用exception隱式對象獲得出錯信息。 JSP引擎會根據page指令的contentType屬性生成相應的調用ServletResponse.setContentType方法的語句。
2)include指令: include指令用於引入其他JSP頁面,如果使用include指令引入了其他JSP頁面,那麼JSP引擎將把這兩個JSP翻譯成一個servlet,所以include指令引入通常稱為靜態引入。由於JSP引擎會把多個JSP頁面翻譯成一個Servlet,所以,在被引入的JSP中,全局架構的代碼可以去掉(<html><head><body>等)(當然這不是必須的),這樣避免與當前JSP頁面中的重複。 語法:
<% include file="relativeURL"%>
其中file屬性用於指定被引入文件的路徑。以"/"開頭,表示代表當前web應用。
幾個細節需要注意: · 被引入的文件必須遵循JSP語法; · 被引入的文件可以使用任意的擴展名,即使擴展名為html,JSP引擎也會按照處理JSP頁面的方式去處理,為了見名知意,JSP規範建議使用.jspf(JSP fragments)作為靜態引入文件的擴展名; · 由於使用include指令將會涉及到2個JSP頁面,並會把2個JSP翻譯成一個Servlet,所以這兩個JSP頁面的指令不能衝突(除了pageEncoding和導包除外)。比如現在要引入兩個JSP頁面,一個JSP中session="true",另一個JSP中session="false",這樣在引入這兩個JSP頁面時就會產生衝突,伺服器會報錯。 在JSP中也可以使用java代碼實現動態包含:
<% request.getRequestDispather("relativeURL").include(request,response);%>
這樣JSP引擎會將不同的JSP頁面翻譯成不同的Servlet,動態包含只是引入其他JSP頁面的結果。動態包含的時候,伺服器會調用多個web資源,而靜態包含時,被翻譯成一個Servlet,伺服器只需要調用一個web資源,所以靜態包含性能好一點,開發時用靜態包含。
3)taglib指令:用來導入自定義標籤庫,見後面自定義標籤部分的內容
5. JSP運行原理和9大隱式對象由上文可知:每個JSP頁面在第一次被訪問時,web容器都會把請求交給JSP引擎(即一個java程序)去處理。JSP引擎現將JSP翻譯成一個_jspServlet(實質上也是一個Servlet),然後按照Servlet的調用方式進行調用。由於JSP第一次訪問時會翻譯成Servlet,所以第一次訪問通常會比較慢,但第二次訪問,JSP引擎如果發現JSP沒有變化,就不再翻譯,而是直接調用,所以程序的執行效率不會受到影響。 JSP引擎在調用JSP對應的_jspServlet時,會傳遞或創建9個與web開發相關的對象供_jspServlet使用。JSP技術的設計者為便於開發人員在編寫JSP頁面時獲得這些web對象的引用,特意定義了9個相應的變數,開發人員在JSP頁面中通過這些變數就可以快速獲得這9大對象的引用,9大隱式對象是哪些以及各自的作用是什麼?
request://就是Servlet里的request response: //就是Servlet里的response session: //就是Servlet里的session application: //就是servlet里的servletContext config: //就是Servlet里的servletConfig page: //就是Servlet自己 exception: //異常,只有errorPage才有 out://JSP頁面輸出 pageContext://pageContext對象是JSP技術中最重要的一個對象,它代表JSP頁面的運行環境
pageContext對象是JSP技術中最重要的一個對象,它代表JSP頁面的運行環境,這個對象不僅封裝了對其它8大隱式對象的引用,它自身還是一個域對象,可以用來保存數據。並且,這個對象還封裝了web開發中經常涉及到的一些常用操作,例如引入和跳轉其它資源、檢索其它域對象中的屬性等。
getException //方法返回exception隱式對象 getPage //方法返回page隱式對象 getRequest() //方法返回request隱式對象 getResponse() //方法返回response隱式對象 getgetServletContext() //方法返回application隱式對象 getServletConfig() //方法返回config隱式對象 getSession() //方法返回session隱式對象 getOut() //方法返回out隱式對象
pageContext作為一個域對象,還封裝了下面的方法:
//pageContext域對象的方法: public void setAttribute(String name, Object value) public Object getAttribute(String name) public void removeAttribute(String name) //pageContext對象中還封裝了訪問其它域的方法(重要) public void setAttribute(String name, Object value, int scope) public Object getAttribute(String name, int scope) public void removeAttribute(String name, int scope) //代表各個域的常量 pageContext.APPLICATION_SCOPE pageContext.SESSION_SCOPE pageContext.REQUEST.SCOPE pageContext.PAGE_SCOPE
到現在為止,應該可以看出pageContext對象的強大之處了!另外還有個findAttribute方法(*重要,查找各個域中的屬性,EL表達式就依賴於這個方法),可以直接調用pageContext.findAttribute(String name),首先會從pageContext里找該屬性,如果沒有,會依次按照下面順序在相應的域中查找:request, session, servletContext,如果沒找著,返回null,否則返回屬性值。 pageContext類中定義了一個forward方法和兩個include方法分別簡化和替代RequestDispatcher.forward方法和include方法,方法接收的資源如果以"/"開頭,"/"代表當前web應用。不過這裡的include是動態包含,不建議使用include。
6. JSP標籤JSP標籤也稱為JSP動作元素,它用於在JSP頁面中提供業務邏輯功能,避免在JSP頁面中直接編寫java代碼造成JSP頁面難以維護。JSP有三種標籤
1) <jsp:include>標籤<jsp:include>標籤用於把另外一個資源的輸出內容插入進當前JSP頁面的輸出內容之中,這種在JSP頁面執行時的引入方式稱為動態引入。 語法:
<jsp:include page="relativeURL | <%= expression%>" flush="true | false" />
相當於調用pageContext.include("relativeURL") page屬性用於指定被引入資源的相對路徑,它也可以通過執行一個表達式來獲得。flush屬性指定在插入其他資源的輸出內容時,是否先將當前JSP頁面都已輸出的內容刷新到客戶端。
2) <jsp:forward>標籤<jsp:forward>標籤用於把請求轉發給另一個資源 語法:
<jsp:forward page="relativeURL | <%= expression%>" />
page屬性用於指定請求轉發到的資源的相對路徑,它可以通過執行一個表達式來獲得。
3) <jsp:param>標籤當使用<jsp:include>和<jsp:forward>標籤引入或將請求轉發給其它資源時,可以使用<jsp:param>標籤向這個資源傳遞參數。 語法1:
<jsp:include page="relativeURL | <%=expression%>"> <jsp:param name="parameterName" value="parameterValue | <%=expression%>" /> </jsp:include>
相當於relativeURL?name=...&value=.... 語法2:
<jsp:forward page="relativeURL | <%=exception%>"> <jsp:param name="parameterName" value="parameterValue | <%=expression%>" /> </jsp:forward>
相當於relativeURL?name=...&value=.... <jsp:param>標籤的name屬性用於指定參數名,value屬性用於指定參數值。在<jsp:include>和<jsp:forward>標籤中可以使用多個<jsp:param>標籤來傳遞多個參數。
7. JSP映射
JSP也可以像Servlet那樣映射,因為JSP本來就是Servlet。
<servlet> <servlet-name>SimpleJspServlet</servlet-name> <jsp-file>/jsp/simple.jsp</jsp-file> </servlet> <servlet-mapping> <servlet-name>SimpleJspServlet</servlet-name> <url-pattern>/xxx/yyy.html</url-pattern> </servlet-mapping>
/jsp/simple.jsp表示在webRoot/jsp目錄下的simple.jsp文件
8.四個域對象到目前為止,web開發共接觸到了4個域對象,這4個域對象是學習web的重點,也是筆試經常考察的知識點:
pageContext(稱為page域): //pageContext中存的數據在頁面範圍都可以取出 request(稱為request域): //request中存的數據在請求範圍內都可以取出 session(稱為session域): //session中存的數據在會話範圍內都可以取出 servletContext(稱為application域)://servletContext中存的數據在整個應用程序範圍內都可以取出
明確如下兩個問題:這四個對象的生命周期?哪種情況下用哪種對象? request:如果客戶機向伺服器發請求產生的數據,用戶看完就沒用了,向這樣的數據就存在request域中。如用戶看的新聞。 session:如果客戶機向伺服器發請求產生的數據,用戶用完了等一會兒還有用,向這樣的數據就存在session域中。如用戶購買的東西,因為結賬還要用到。 servletContext:如果客戶機向伺服器發請求產生的數據,用戶用完了還要給其他用戶用,向這樣的數據就存在servletContext域中。如聊天室中說出的話,因為話要在自己頁面中看到,別人也要看到。 實際中,能用小的容器就不要用大的,即request能滿足就不要用session,session能滿足就不要用servletContext
9.總結由於JSP一般都是通過servlet轉發過來,servlet會通過容器將數據帶過來,所以會使用JSP從容器中取出數據然後顯示出來即可。取數據用腳本片段<%%>來取,顯示用腳本表達式<%= %>處理。在學習了EL表達式和自定義標籤後,腳本片段就可以用自定義標籤替代,腳本表達式就可以用EL表達式替代了。
推薦閱讀:
※我是一名佛系整理師
※所謂「絕招」都從義理總結中得來
※中國歷代禁書題解(詩涇)孔子刪定編纂(我國第一部詩歌總結)
※乒乓球實用技巧 幾種非常好用的弧圈球技術動作總結-乒器店
※各大半導體廠商ADAS技術解決方案總結