我們都知道在JSP中include有兩種形式,分别是
<%@ include file=” ”%>
<jsp:include page=” ” flush=”true”/>
前者是指令元素,後者是動作元素。具體它們将在何處用?如何用及它們有什麼差別?這應該是很多人看到它都會想到的問題。下面一起來看看吧。
通常當應用程式中所有的頁面的某些部分(例如标題、頁腳和導航欄)都相同的時候,我們就可以考慮用include。具體在哪些時候 用<%@ include file=” ”%>,哪些時候用< jsp:include page=” ” flush=”true”/>.這種形式。首先要明白的是它們之間的差別。隻有了解了它們用法的不同才 了解該在何時去用以及如何選擇。
<%@ include file=” ”%>,jsp的include指令元素讀入指定頁面的内容。并把這些内容和原來的頁面融合到一起。(這個過程是在翻譯階段:也就是jsp被轉化成servlet的階段進行的。
這裡對翻譯階段進行一下說明:我們知道,jsp頁面不能原封不動地被傳送給浏覽器,所有的jsp元素都必須首先由伺服器進行處理。這是通過将jsp頁面轉達化成servlet,然後執行這個servlet來完成的。伺服器需要一個jsp容器來處理jsp頁面。jsp容器通常以servlet的形式來實作,這個servlet經過配置,可以處理對jsp頁面的所有請求。
Jsp容器負責将jsp頁面轉化成servlet(稱為jsp頁面實作類?JSP Page implementation class),并編譯這個servlet。這兩步就構成了翻譯階段.
由此我們就會知道:jsp頁面是把include指令元素所指定的頁面的實際内容(也就是代碼段)加入到引入它的jsp頁面中,合成一個檔案後被jsp容器将它轉化成servlet。可以看到這時會産生一個臨時class檔案和一個java檔案。下面舉個例子。
伺服器用tomcat,引入頁面的jsp檔案叫test.jsp。被引入的頁面叫date.jsp.這個jsp檔案裡存放的是一個有關時間的jsp代碼,目前的上下文根設為test
date.jsp的源檔案:
|
以下是test.jsp的源檔案:
|
在test.jsp 檔案中,我們隻輸出了一行文本“ 有關jsp中include的兩種用法.敬請關注。”,現在讓我們先 用<%@ include file=”date.jsp” %>這種形式引入date.jsp這個檔案。你想會出現什麼問題了嗎?此時出現 了錯誤提示:
|
以下還有一堆錯誤,但我們隻要看這裡就知道問題的所在了。狀态碼為http 500伺服器内部錯誤。再看下面的提示。在date.jsp頁面中不能指定多個contentType.
原因就在這裡了。是因為在翻譯階段,date.jsp檔案的代碼被原封不動地加入到了test.jsp頁面進而合成一個檔案。合成後的檔案中就會相同的:
|
這句代碼。解決的辦法是把date.jsp檔案中的這句删掉,重新整理後再請求test.jsp頁面。
請求test.jsp在頁面顯示如下:
|
這時我們還不能發現什麼。還是去檢視tomcat下的臨時檔案吧。到那裡去看看date.jsp檔案的内容是否已被加入到了test.jsp檔案中。
<注.此處的tomcat裝在E盤根目錄下>
目錄
E:\tomcat\work\Standalone\localhost\test.
在這個目錄下會看到
test_jsp.java和test_jsp.class兩個檔案。
這裡的java檔案就是jsp容器将jsp轉化成了servlet而得到的test_jsp.java這個檔案。
相對應的test_jsp.class這個檔案就是編譯test_jsp.java這個servlet檔案産生的類檔案了。打開所産生的servlet文 件(test_jsp.java)。此時我們會發現,在test.jsp 檔案被轉化成servlet檔案時,在輸出的<haed>之間加入 了一些不是test.jsp頁面裡面的代碼,新加入的内容就是 date.jsp裡面的代碼: 新加入了哪些内容或是否真的加入了新的内容請自己測試去看 一下就會一目了然了.在這裡不再詳述.
以上就是我們用<%@ include file=”date.jsp”%>這種形式得到的結果.
下面我們換用<jsp:include page=”dae.jsp” flush=”true”/>也就是将<%@ include file=”date.jsp”%>換成<jsp:include page=”dae.jsp” flush=”true”/>,然後請求test.jsp.
|
此時會在頁面上看見.我們所引入date.jsp輸出的日期中中文出現了亂碼.什麼原因?是因為include行為元素是在請求處理階段執行的(此處要對請求處理階段進行說明一下,Jsp容器除了上面提到的負責将jsp頁面轉化成servlet外,還負責調用jsp頁面實作類以處理每個請求并産生應答.這個階段我們就稱為請求處理階段.請求處理階段僅執行類檔案)。
是以在我們作include行為元素引入頁面時,實際隻是引用了date.jsp這個檔案被轉化并被編譯後産生的servlet類檔案.既如此, date.jsp就是作為一個單獨的檔案在執行後才被test.jsp檔案運作時調用.由于date.jsp檔案中沒有指定字元編碼.是以出現了亂 碼.解決辦法是在date.jsp檔案中重新把剛才去掉的
|
這行語句加入後重新整理重新運作.此時頁面顯示正确,并跟用include指令正常運作時一樣.再檢視tomcat下的臨時檔案會發現.此時多出了一個 date_jsp.java檔案和一個date_jsp.class檔案.這兩個檔案得來的方式跟test_jsp.java 和 test_jsp.class檔案得來的方式一樣.再檢視一下此時test_jsp.java檔案的代碼會發現.此時隻新增加了一句代碼:
|
它并沒有把date.jsp檔案的代碼加入到test.jsp.
隻是在運作時引入了date.jsp頁面執行後所産生的應答.這意味着我們可以指定任何能夠産生應答的Web資源,(例如一個servlet或一個jsp頁面),隻要這些資源所産生的類型和jsp頁面産生的内容類型相同.JSP容器将通過一個内部的函數調用來執行指定的資源.是以,這些被引入的資源可以幫 助處理原來的請求,是以這些資源可以通路請求作用域内的所有對象.以及所有原來的請求參數.
由于在首頁面被請求時,這些頁面還沒有被引入到首頁面中,是以你可以對page屬性使用一個請求時屬性值,以便根據運作時的情況來決定要引入哪一個頁面.還可以添加一些将被引入的頁面讀取的請求參數.
|
如果修改了被引入的jsp頁面,那麼可以立刻使用該頁面的最新版本,這是因為對待被引入的頁面的方式與對待由浏覽器直接調用的jsp頁面的方式完全相同.即容器檢測頁面的變化,并自動進入翻譯階段,以得到頁面的最新版本.
注意:include動作元素同jsp其它元素一樣,沒有行為體時要以”/”結束.就像下面這樣.<jsp:include page=”<%=pageSelectedAtRuntime%>” flush=”true” />
以下是對include 兩種用法的差別,主要有兩個方面的不同:
執行時間上:
<%@ include file=”relativeURI”%> 是在翻譯階段執行
<jsp:include page=”relativeURI” flush=”true” /> 在請求處理階段執行.
引入内容的不同:
<%@ include file=”relativeURI”%>引入靜态文本(html,jsp),在JSP頁面被轉化成servlet之前和它融和到一起.
<jsp:include page=”relativeURI” flush=”true” />引入執行頁面或servlet所生成的應答文本.
另外在兩種用法中file和page屬性都被解釋為一個相對的URI.如果它以斜杠開頭,那麼它就是一個環境相關的路徑.将根據賦給應用程式的URI的字首進行解釋,如果它不是以斜杠開頭,那麼就是頁面相關的路徑,就根據引入這個檔案的頁面所在的路徑進行解釋。
原文連結:
http://developer.51cto.com/art/200902/111134_1.htm