[Servlet] <%@ include file=""%>& 區別
我們都知道在jsp中包括有兩種形式,分別是包括指令:
<%@ include file =“”%>和include動作:<jsp:include page =“”flush =“true”/>
前者是指令元素,後者是行為元素。具體它們將在何處用?如何用及它們有什麼區別?這應該是很多人看到它都會想到的問題。下面一起來看看吧。
通常當應用程序中所有的頁面的某些部分(例如標題,頁腳和導航欄)都相同的時候,我們就可以考慮用包括。具體在哪些時候用<%@ include file =“”%>,哪些時候用<jsp:include page =“”flush =“true”/>這種形式。首先要明白的是它們之間的區別。只有了解了它們用法的不同才理解該在何時去用以及如何選擇。
前者是指令元素,後者是行為元素。具體它們將在何處用?如何用及它們有什麼區別?這應該是很多人看到它都會想到的問題。下面一起來看看吧。
通常當應用程序中所有的頁面的某些部分(例如標題,頁腳和導航欄)都相同的時候,我們就可以考慮用包括。具體在哪些時候用<%@ include file =“”%>,哪些時候用<jsp:include page =“”flush =“true”/>這種形式。首先要明白的是它們之間的區別。只有了解了它們用法的不同才理解該在何時去用以及如何選擇。
兩者最重要的區別:JSP指令<%@ include file =“”%>,是將被引入的JSP與原JSP融合到一起,而這個融合過程在翻譯階段進行的。
為什麼需要翻譯階段?我們知道,JSP頁面並不是原封不動的發送到客戶端的,因為瀏覽JSP頁面的客戶端並不需要安裝的Java虛擬機,客戶端機器並不能讀懂JSP,它能讀懂的只有HTML,JavaScript(當然還有其他,例如:Applet,Flex,AxtiveX等等,但那些都需要下載相應的客戶端解析器),這樣就需要Servlet Engine(例如:Tomcat)將所有的JSP元素進行處理。這是通過將JSP頁面轉化成Servlet中,然後執行這個的Servlet來完成的。服務器需要一個jsp的容器來處理JSP頁面的.jsp容器通常以的Servlet的形式實現,這個小服務程序經過配置,可以處理對JSP頁面的所有請求。
JSP容器負責將JSP頁面轉化成servlet中,並編譯這個servlet的。這兩個步驟就構成了翻譯階段。
而JSP翻譯之後的servlet的輸出的內容才是客戶端瀏覽器能夠識別的東西,HTML,JavaScript的之類的,servlet的是使用的JspWriter對象輸出輸出這些HTML,JavaScript的的。如果你去翻看翻看JSP編譯後的小服務程序代碼,你會發現很有意思的東西,比如Struts的的<邏輯:迭代>標籤,被翻譯成做{}而()語句實現循環如果我們把<豆:寫>。寫在<邏輯:迭代>內部,則在做內部會出現類似_jspx_meth_bean_write_2(_jspx_th_logic_iterate_0,_jspx_page_context)的方法調用。這些說明了一切。
由此我們知道:jsp頁面是把包括指令元素(<%@ include file =“”%>)所指定的頁面的實際內容(也就是代碼段)加入到引入它的jsp頁面中,合成一個文件被jsp容器將它轉化成servlet。可以看到這時會產生一個臨時類文件和一個servlet源文件。而動作元素(<jsp:include page =“”/>)是在請求處理階段引入的,會被JSP容器生成兩個臨時類文件和兩個servlet的原文件。而引入的只是小服務程序的輸出結果,即的JspWriter對象的輸出結果,而不是JSP的源代碼。
舉個例子:
main.jsp中
- < %@ page language = “java” pageEncoding = “GBK” %>
- < %@ taglib uri = “http://struts.apache.org/tags-bean” prefix = “bean” %>
- < %@ taglib uri = “http://struts.apache.org/tags-html” prefix = “html” %>
- < %@ taglib uri = “http://struts.apache.org/tags-logic” prefix = “logic” %>
- < %@ taglib uri = “http://struts.apache.org/tags-tiles” prefix = “tiles” %>
- <!DOCTYPE HTML PUBLIC“ - // W3C // DTD HTML 4.01 Transitional // EN” >
- < html:html lang = “true” >
- < head >
- < html:base />
- < title > index.jsp </ title >
- < meta http-equiv = “Content-Type” content = “text / html; charset = GBK” />
- < meta http-equiv = “pragma” content = “no-cache” >
- < meta http-equiv = “cache-control” content = “no-cache” >
- < meta http-equiv = “expires” content = “0” >
- < meta http-equiv = “keywords” content = “keyword1,keyword2,keyword3” >
- < meta http-equiv = “description” content = “這是我的頁面” >
- <! -
- <link rel =“stylesheet”type =“text / css”href =“styles.css”>
- - >
- </ head >
- < body >
- <! - 標題頁信息 - >
- < % - < %@ include file = “include / head.jsp” %> - %>
- < jsp:include page = “include / head.jsp” > </ jsp:include >
- <! - 導航欄 - >
- < % - < %@ include file = “include / menubar.jsp” %> - %>
- < jsp:include page = “include / menubar.jsp” > </ jsp:include >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- 我是主體< BR > < BR > < BR >
- < % - < %@ include file = “include / copyright.jsp” %> - %>
- < % - < jsp:include page = “ / includeSample_copyright.do” > </ jsp:include > - %>
- < % - < jsp:include flush = “true” page = “include / copyright.jsp” > </ jsp:include > - %>
- </ body >
- </ html:html >
head.jsp
- < % - < %@ page language = “java” pageEncoding = “GBK” %> - %>
- < % - < %@ taglib uri = “/WEB-INF/struts-html.tld” prefix = “html” %> - %>
- < % - < %@ taglib uri = “/WEB-INF/struts-logic.tld” prefix = “logic” %> - %>
- < % - < %@ taglib uri = “/WEB-INF/struts-bean.tld” prefix = “bean” %> - %>
- < table >
- < tr >
- < td > < bean:message key = “copyright.inc.copyright” />
- 我是頭我是頭我是頭我是頭我是頭我是頭我是頭</ td >
- </ tr >
- </ table >
以head.jsp為例:
1,如果我是用<%@ include file =“include / head.jsp”%>引入,注意,head.jsp被我註釋掉的字符集和Struts標籤的引入,如果打開註釋,會怎麼樣呢? ?會拋出500異常,/ main.jsp(44,4)/include/head.jsp ( 3,56)嘗試將前綴html重新定義到/WEB-INF/struts-html.tld,當它已經被定義作為http://struts.apache.org/tags-html在當前的範圍。這時因為在翻譯階段main.jsp和head.jsp被原封不動的合稱為一個jsp,察看Tomcat工作目錄只有一個servlet類文件。
\工作\卡塔利娜\本地主機\ IncludeAction \組織\阿帕奇\ JSP \ main_jsp.java
\工作\卡塔利娜\本地主機\ IncludeAction \組織\阿帕奇\ JSP \ main_jsp.class
試想,在一個類文件中兩次引入相同的Struts的標籤,編譯時當然回拋出異常了。
試想,在一個類文件中兩次引入相同的Struts的標籤,編譯時當然回拋出異常了。
2,那麼如果我是以<jsp:include page =“include / head.jsp”/>方式引入呢?首先先將head.jap中註釋的Struts標籤的部分打開,而字符集部分仍然註釋結果出現亂碼:
- 結果:
- 2004 - 2006 版權所有
- 頭?頭?頭?頭?頭?頭?頭?頭?
- ????菜單欄
- 我是主體?
什麼原因?包括動作元素是在請求階段執行引入的,所以它引入的只是head.jsp被翻譯成的servlet文件中_jspService這個方法中的JspWriter這個對象的輸出(out.write()方法的輸出流)。該輸出的執行是在head.jsp被引入main.jsp中之前就進行了,所以main.jsp中頁面中的字符集設置當然對head.jsp不起作用了。
如果Struts標籤部分也註釋掉呢?“2004-2006版權所有”這一行不會輸出,因為這一行是由Struts標籤輸出的,沒有輸出的原因和字符集相同,我想大家應該明白了。
最後觀察Tomcat的工作目錄下,會有兩個Servlet的:
第一個:\工作\卡塔利娜\本地主機\ IncludeAction \組織\阿帕奇\ JSP \ main_jsp.java
\工作\卡塔利娜\本地主機\ IncludeAction \組織\阿帕奇\ JSP \ main_jsp.class
第二個:\工作\卡塔利娜\本地主機\ IncludeAction \組織\阿帕奇\ JSP \ \包括head_jsp.java
\工作\卡塔利娜\本地主機\ IncludeAction \組織\阿帕奇\ JSP \ \包括head_jsp.class
另外,如果希望通過修改後綴的方法表示哪些是被引入的文件,例如:將head.jsp改名為head.inc的話, JSP容器不能識別* .inc。所以不能翻譯head.inc,所以此時只能使用<%@ include file =“head.inc”%>方法引入一個文件了。
另外,如果希望通過修改後綴的方法表示哪些是被引入的文件,例如:將head.jsp改名為head.inc的話, JSP容器不能識別* .inc。所以不能翻譯head.inc,所以此時只能使用<%@ include file =“head.inc”%>方法引入一個文件了。
我們來總結一下兩種包括兩種用法的區別,主要有兩個方面的不同:
一,執行時間上:
<%@ include file =“relativeURI”%>是在翻譯階段執行
<jsp:include page = relativeURI“沖水=”真“/>在請求處理階段執行。
二,二:引入內容的不同:
<%@包括文件=”relativeURI“%>引入靜態文本(HTML,JSP),在JSP頁面被轉化成servlet之前和它融合到一起。
<jsp:include page =“relativeURI”flush =“true”/>引入執行頁面或servlet所生成的應答文本。
一,執行時間上:
<%@ include file =“relativeURI”%>是在翻譯階段執行
<jsp:include page = relativeURI“沖水=”真“/>在請求處理階段執行。
二,二:引入內容的不同:
<%@包括文件=”relativeURI“%>引入靜態文本(HTML,JSP),在JSP頁面被轉化成servlet之前和它融合到一起。
<jsp:include page =“relativeURI”flush =“true”/>引入執行頁面或servlet所生成的應答文本。
留言
張貼留言