我們都知道在jsp中include有兩種形式,分別是
<%@ include file=” ”%>
<jsp:include page=” ” flush=”true”/></P>
前者是指令元素、後者是行為元素。具體它們將在何處用?如何用及它們有什麼差別?這應該是很多人看到它都會想到的問題。下面一起來看看吧。 </P>
通常當應用程式中所有的頁面的某些部分(例如標題、頁尾和導覽列)都相同的時候,我們就可以考慮用include。具體在哪些時候用<%@ include file=” ”%>,哪些時候用<jsp:include page=” ” flush=”true”/>。這種形式。首先要明白的是它們之間的差異。只有了解了它們用法的不同才理解該在何時去用以及如何選擇。
<%@ include file=” ”%>,jsp的include指令元素讀入指定頁面的內容。並把這些內容和原來的頁面融合在一起。 (這個過程是在翻譯階段:也就是jsp被轉化成servlet的階段進行的。</P>
這裡對翻譯階段進行一下說明:我們知道,jsp頁面不能原封不動地被傳送給瀏覽器,所有的jsp元素都必須先由伺服器處理。這是透過將jsp頁面轉達化成servlet,然後執行這個servlet來完成的。伺服器需要一個jsp容器來處理jsp頁面。 jsp容器通常以servlet的形式來實現,這個servlet經過配置,可以處理對jsp頁面的所有請求。 </P>
Jsp容器負責將jsp頁面轉換成servlet(稱為jsp頁面實作類別?JSP Page implementation class),並編譯這個servlet。這兩步就構成了翻譯階段.</P>
由此我們就會知道:jsp頁面是把include指令元素所指定的頁面的實際內容(也就是程式碼段)加入到引入它的jsp頁面中,合成一個檔案後被jsp容器將它轉換成servlet。可以看到這時會產生一個臨時class檔案和一個java檔。下面舉個例子。 </P>
伺服器用tomcat,引入頁面的jsp檔案叫test.jsp。被引入的頁面叫date.jsp.這個jsp檔案裡存放的是一個有關時間的jsp程式碼,當前的上下文根設為test
//======date.jsp的來源檔案=====//
<%@ page language=”java” contentType="text/html;charset=gb2312"%>
<%
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這個檔.
<head>
<body>
<table><tr><td>
有關jsp中include的兩種用法.敬請關注。
</td></tr></table>
</body>
</html></P>
在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.</P>
原因就在這裡了。是因為在翻譯階段,date.jsp檔案的程式碼被原封不動地加入到了test.jsp頁面從而合成一個檔案。合成後的文件中就會相同的:
<%@ page language=”java” contentType=”text/html;charset=gb2312”%>
這句代碼。解決的方法就是把date.jsp檔案中的這句話刪掉。刷新後再請求test.jsp頁面</P>
請求test.jsp在頁面顯示如下
2003年12月10日13:12:40
有關jsp中include的兩種用法.敬請關注。 </P>
這時我們還不能發現什麼。還是去查看tomcat下的臨時檔案吧。到那裡去看看date.jsp檔案的內容是否已加入了test.jsp檔案。
<註.此處的tomcat裝在E盤根目錄下>
目錄
E:tomcatworkStandalonelocalhosttest.
在這個目錄下會看到
test_jsp.java和test_jsp.class兩個檔案。 </P>
這裡的java檔案就是jsp容器將jsp轉換成了servlet而得到的test_jsp.java這個檔。 </P>
相對應的test_jsp.class這個檔案就是編譯test_jsp.java這個servlet檔案產生的類別檔案了。開啟所產生的servlet檔案(test_jsp.java)。此時我們會發現,在test.jsp 檔案被轉換成servlet檔案時,在輸出的<haed>之間加入了一些不是test.jsp頁裡面的程式碼,新加入的內容就是date.jsp裡面的程式碼:新加入了哪些內容或是否真的加入了新的內容請自己測試去看一下就會一目了然了.在這裡不再詳述.</P>
以上就是我們用<%@ 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
有關jsp中include的兩種用法.敬請關注。 </P>
此時會在頁面上看見.我們所引入date.jsp輸出的日期中中文出現了亂碼.什麼原因?是因為include行為元素是在請求處理階段執行的(此處要對請求處理階段進行說明一下. Jsp容器除了上面提到的負責將jsp頁面轉換成servlet外,還負責調用jsp頁面實現類以處理每個請求並產生應答.這個階段我們就稱為請求處理階段.請求處理階段僅執行類文件) 。 </P>
所以在我們作include行為元素引入頁面時,實際上只是引用了date.jsp這個文件被轉化並被編譯後產生的servlet類文件.既如此, date.jsp就是作為一個單獨的文件在執行後才被test .jsp檔案運行時呼叫.由於date.jsp檔案中沒有指定字元編碼.所以出現了亂碼.解決辦法是在date.jsp檔案中重新把剛才去掉的
<%@ page language=”java” contentType=”text/html;charset=gb2312”%>
這行語句加入後刷新重新運行.此時頁面顯示正確,並跟用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);</P>
它並沒有把date.jsp檔的程式碼加入test.jsp.</P>
只是在運行時引入了date.jsp頁面執行後所產生的應答.這意味著我們可以指定任何能夠產生應答的Web資源,(例如一個servlet或一個jsp頁面),只要這些資源所產生的類型和jsp頁面產生的內容類型相同.JSP容器將透過一個內部的函數呼叫來執行指定的資源.因此,這些被引入的資源可以幫助處理原來的請求,所以這些資源可以存取請求作用域內的所有物件.以及所有原來的請求參數.</P>
由於在主頁面被請求時,這些頁面還沒有被引入到主頁面中,所以你可以對page屬性使用一個請求時屬性值,以便根據運行時的情況來決定要引入哪一個頁面.還可以添加一些將被引入的頁面讀取的請求參數.
<jsp:include page=”<%=pageSelectedAtRuntime%>” flush=”true” >
<jsp:param name=”fitstParamer” value=”firstvalue”>
<jsp:param name=”lastParamer” value=”lastvalue”>
</jsp:include></P>
如果修改了被引入的jsp頁面,那麼可以立刻使用該頁面的最新版本,這是因為對待被引入的頁面的方式與對待由瀏覽器直接調用的jsp頁面的方式完全相同.即容器檢測頁面的變化,並自動進入翻譯階段,以得到頁面的最新版本.</P>
(注意,include行為元素同jsp其它元素一樣,沒有行為體時要以”/”結束.就像下面這樣.
<jsp:include page=”<%=pageSelectedAtRuntime%>” flush=”true” />)</P>
以下是對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的前綴進行解釋,如果它不是以斜槓開頭,那麼就是頁面相關的路徑,就根據引入這個文件的頁面所在的路徑進行解釋.更多有關URL是如何解釋的請參考相關資料或書籍.