天天看點

include與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的源檔案=====// 

<%@ page language=”java” c%> 

<% 

java.util.date date=new java.util.date(); 

string date_cn =""; 

string datestr = ""; 

switch(date.getday()) 

case 0:date_cn ="日"; break; 

case 1:date_cn ="一"; break; 

case 2:date_cn ="二"; break; 

case 3:date_cn ="三"; break; 

case 4:date_cn ="四"; break; 

case 5:date_cn ="五"; break; 

case 6:date_cn ="六"; break; 

datestr = (1900+date.getyear()) + "年" + (date.getmonth()+1) + "月" + date.getdate() + "日(星期" + date_cn + ")"; 

%> 

document.write("<%=datestr%>"); 

//======以下是test.jsp的源檔案=============// 

<%@ page language=”java” contenttype=”text/html;charset=gb2312”%> 

<html> 

<head> 

<title>include的兩種用法</title> 

<jsp:include page=”date.jsp” flush=”true”/> 

<%--@ include file=”date.jsp” %--> 

//我們在這裡用include的兩種不同形式來引入date.jsp這個檔案. 

<body> 

<table><tr><td> 

有關jsp中include的兩種用法.敬請關注。 

</td></tr></table> 

</body> 

</html> 

在test.jsp 檔案中,我們隻輸出了一行文本“ 有關jsp中include的兩種用法.敬請關注。”,現在讓我們先用<%@ include file=”date.jsp” %>這種形式引入date.jsp這個檔案。你想會出現什麼問題了嗎?此時出現了錯誤提示: 

http status 500 ? 

org.apache.jasper.jasperexception: /date.jsp(0,0) page directive: can't have multiple occurrences of contenttype 

以下還有一堆錯誤,但我們隻要看這裡就知道問題的所在了。狀态碼為http 500伺服器内部錯誤。再看下面的提示。在date.jsp頁面中不能指定多個contenttype. 原因就在這裡了。是因為在翻譯階段,date.jsp檔案的代碼被原封不動地加入到了test.jsp頁面進而合成一個檔案。合成後的檔案中就會相同的: 

這句代碼。解決的辦法是把date.jsp檔案中的這句删掉。重新整理後再請求test.jsp頁面 

請求test.jsp在頁面顯示如下 

2003年12月10日 13:12:40 

這時我們還不能發現什麼。還是去檢視tomcat下的臨時檔案吧。到那裡去看看date.jsp檔案的内容是否已被加入到了test.jsp檔案中。 

在目錄下會看到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. 

2003? ê 12??10?? 13:30:13 

此時會在頁面上看見.我們所引入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檔案的代碼會發現.此時隻新增加了一句代碼: 

jspruntimelibrary.include(request, response, "date.jsp", out, true); 

它并沒有把date.jsp檔案的代碼加入到test.jsp.隻是在運作時引入了date.jsp頁面執行後所産生的應答.這意味着我們可以指定任何能 夠産生應答的web資源,(例如一個servlet或一個jsp頁面),隻要這些資源所産生的類型和jsp頁面産生的内容類型相同.jsp容器将通過一個 内部的函數調用來執行指定的資源.是以,這些被引入的資源可以幫助處理原來的請求,是以這些資源可以通路請求作用域内的所有對象.以及所有原來的請求參 數. 

由于在首頁面被請求時,這些頁面還沒有被引入到首頁面中,是以你可以對page屬性使用一個請求時屬性值,以便根據運作時的情況來決定要引入哪一個頁面.還可以添加一些将被引入的頁面讀取的請求參數. 

<jsp:include page=”<%=pageselectedatruntime%>” flush=”true” > 

<jsp:param name=”fitstparamer” value=”firstvalue”> 

<jsp:param name=”lastparamer” value=”lastvalue”> 

</jsp:include> 

如果修改了被引入的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的字首進行解釋,如果它不是以斜杠開頭,那麼就是頁面相關的路徑,就根據引入這個檔案的頁面所在的路徑進行解釋.