咱們都知道在jsp中include有兩種形式,分別是Include指令:<%@ include file=""%>
和include動做:<jsp:include page="" flush="true"/>
。前者是指令元素、後者是行爲元素。具體它們將在何處用?如何用及它們有什麼區別?這應該是不少人看到它都會想到的問題。下面一塊兒來看看吧。css
一般當應用程序中全部的頁面的某些部分(例如標題、頁腳和導航欄)都相同的時候,咱們就能夠考慮用include。具體在哪些時候用<%@ include file=""%>
,哪些時候用<jsp:include page="" flush="true"/>
。這種形式。首先要明白的是它們之間的區別。只有瞭解了它們用法的不一樣才理解該在什麼時候去用以及如何選擇。html
二者最重要的區別:JSP指令<%@ include file=""%>
,是將被引入的JSP與原JSP融合到一塊兒,而這個融合過程是在翻譯階段進行的。java
爲何須要翻譯階段?咱們知道,jsp頁面並非原封不動的發送到客戶端的,由於瀏覽JSP頁面的客戶端並不須要安裝Java虛擬機,客戶端機器並不能讀懂JSP,它能讀懂的只有HTML、JavaScript (固然還有其餘,例如:Applet、Flex、AxtiveX等等,但那些都須要下載相應的客戶端解析器),這樣就須要Servlet Engine (例如:Tomcat) 將全部的JSP元素進行處理。這是經過將jsp頁面轉化成Servlet,而後執行這個Servlet來完成的。服務器須要一個jsp容器來處理jsp頁面。jsp容器一般以Servlet的形式實現,這個servlet通過配置,能夠處理對jsp頁面的全部請求。apache
JSP容器負責將jsp頁面轉化成servlet,並編譯這個servlet。這兩個步驟就構成了翻譯階段。瀏覽器
而jsp翻譯以後的servlet輸出的內容纔是客戶端瀏覽器可以識別的東西,HTML、JavaScript之類的,servlet是使用JspWriter對象輸出輸出這些HTML、JavaScript的。若是你去翻看翻看jsp編譯後的servlet代碼,你會發現頗有意思的東西,好比Struts的<logic:iterator>
標籤,被翻譯成do{}while()
語句實現循環。若是咱們把<bean:write>
寫在<logic:iterator>
內部,則在do的內部會出現相似_jspx_meth_bean_write_2(_jspx_th_logic_iterate_0, _jspx_page_context)
的方法調用。這些說明了一切。服務器
由此咱們知道:jsp頁面是把include指令元素(<%@ include file=""%>
)所指定的頁面的實際內容(也就是代碼段)加入到引入它的jsp頁面中,合成一個文件後被jsp容器將它轉化成servlet。能夠看到這時會產生一個臨時class文件和一個servlet源文件。而動做元素(<jsp:include page=""/>
)是在請求處理階段引入的,會被JSP容器生成兩個臨時class文件和兩個servlet原文件。而引入的只是servlet的輸出結果,即JspWriter對象的輸出結果,而不是jsp的源代碼。jsp
舉個例子: main.jsp:ui
<%@ 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="This is my page"> <!-- <link rel="stylesheet" type="text/css" href="styles.css"> --> </head> <body> <!-- Header Page Information --> <%-- <%@ include file="include/head.jsp"%>--%> <jsp:include page="include/head.jsp"></jsp:include> <!-- Nav Bar --> <%-- <%@ 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"/> 我是Head 我是Head我是Head我是Head我是Head我是Head我是Head</td> </tr> </table>
以head.jsp爲例:code
<%@ include file="include/head.jsp"%>
引入,注意,head.jsp被我註釋掉的字符集和Struts標籤的引入,若是打開註釋,會怎麼樣呢??會拋出500異常:/main.jsp(44,4) /include/head.jsp(3,56) Attempt to redefine the prefix html to /WEB-INF/struts-html.tld, when it was already defined as http://struts.apache.org/tags-html in the current scope.
這時由於在翻譯階段main.jsp和head.jsp被原封不動的合稱爲一個jsp,察看Tomcat工做目錄只有一個servlet類文件:
\work\Catalina\localhost\IncludeAction\org\apache\jsp\main_jsp.java \work\Catalina\localhost\IncludeAction\org\apache\jsp\main_jsp.class
試想,在一個類文件中兩次引入相同的Struts標籤,編譯時固然回拋出異常了。 2. 那麼若是我是以<jsp:include page="include/head.jsp"/>
方式引入呢?首先先將head.jap中註釋的Struts標籤的部分打開,而字符集部分仍然註釋。結果出現亂碼:
結果: 2004-2006 版權全部 ????Head ????Head????Head????Head????Head????Head????Head ????MenuBar 我是主體...
什麼緣由?include動做元素是在請求階段執行引入的,因此它引入的只是head.jsp被翻譯成servlet文件中_jspService這個方法中JspWriter這個對象的輸出(out.write()方法的輸出流)。該輸出的執行是在head.jsp被引入main.jsp以前就進行了,因此main.jsp頁面中的字符集設置固然對head.jsp不起做用了。
若是Struts標籤部分也註釋掉呢?"2004-2006 版權全部"這一行不會輸出,由於這一行是由Struts標籤輸出的,沒有輸出的緣由和字符集相同,我想你們應該明白了。
最後觀察Tomcat工做目錄下,會有兩個Servlet:
第一個: \work\Catalina\localhost\IncludeAction\org\apache\jsp\main_jsp.java \work\Catalina\localhost\IncludeAction\org\apache\jsp\main_jsp.class 第二個: \work\Catalina\localhost\IncludeAction\org\apache\jsp\include\head_jsp.java \work\Catalina\localhost\IncludeAction\org\apache\jsp\include\head_jsp.class
另外,若是但願經過修改後綴的方法表示哪些是被引入的文件,例如:將head.jsp更名爲head.inc的話,JSP容器不能識別*.inc。因此不能翻譯head.inc,因此此時只能使用<%@ include file="head.inc"%>
方法引入一個文件了。
咱們來總結一下兩種include 兩種用法的區別,主要有兩個方面的不一樣:
執行時間上:
<%@ include file=」relativeURI」%>
是在翻譯階段執行<jsp:include page=」relativeURI」 flush=」true」 />
在請求處理階段執行。 引入內容的不一樣:<%@ include file=」relativeURI」%>
引入靜態文本(html,jsp),在JSP頁面被轉化成servlet以前和它融和到一塊兒。<jsp:include page=」relativeURI」 flush=」true」 />
引入執行頁面或servlet所生成的應答文本。<jsp:include page=」relativeURI?a=1&b=2」 flush=」true」 />
能夠攜帶參數