Java Web中的include

上周在工作上遇到了一件糗事。在寫jsp的過(guò)程中,經(jīng)常會(huì)遇到將jsp文件模塊化,然后通過(guò)include的方式將一個(gè)個(gè)模塊進(jìn)行組裝。這個(gè)過(guò)程中,經(jīng)常會(huì)在各個(gè)模塊之間共享變量,那么變量的范圍是什么呢?自己因?yàn)橥涍@個(gè)知識(shí)點(diǎn)而調(diào)試了好長(zhǎng)時(shí)間。知識(shí)點(diǎn)雖然簡(jiǎn)單,但是這個(gè)是Java web經(jīng)典的問(wèn)題,即對(duì)于include,Java web到底是怎么實(shí)現(xiàn)的?

jsp文件到底是什么?

jsp-servlet-tomcat.png
  • JSP是Servlet的一種特殊形式,每個(gè)JSP頁(yè)面就是一個(gè)Servlet實(shí)例
  • JSP頁(yè)面由系統(tǒng)編譯成Servlet,Servlet再負(fù)責(zé)響應(yīng)用戶請(qǐng)求
  • Tomcat負(fù)責(zé)執(zhí)行Servlet文件

準(zhǔn)備實(shí)驗(yàn)

因此在我機(jī)器上實(shí)驗(yàn)了一把,實(shí)驗(yàn)的環(huán)境如下:

名稱 配置
系統(tǒng) Mac OS
IDE IntelliJ IDEA 2017.2 Help
容器 Tomcat 7.04

項(xiàng)目的文件路徑如下所示:

項(xiàng)目路徑.png

在Mac OS下,Intellij IDEA將jsp文件編譯成相應(yīng)的servlet文件,那么該文件的位置是:
/Users/garybhwang/Library/Caches/IntelliJIdea2017.2/tomcat
然后根據(jù)相應(yīng)的項(xiàng)目名稱找到相應(yīng)的java文件:
/Unnamed_項(xiàng)目名/work/Catalina/localhost/_/org/apache/jsp
相應(yīng)的目錄結(jié)構(gòu)如下所示:

jsp-servlet目錄.png

實(shí)驗(yàn)方式

common.jsp 頁(yè)面去包含header.jsp 頁(yè)面,header.jsp 代碼如下:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>header</title>
</head>
<body>
    <h2>This is header page!</h2>
</body>
</html>

include三種方式

  • jsp頁(yè)面的include指令
    <%@ include file="/WEB-INF/header.jsp"%>
    common.jsp 頁(yè)面的代碼如下:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>common</title>
</head>
<body>
    <h1>This is common page!</h1>
    <%@ include file="/WEB-INF/header.jsp"%>
</body>
</html>

那么common.jsp 編譯成為servlet文件只有的內(nèi)容如下:

(common_jsp.java).png

從上面的源碼可以看到,<%@ include file = "*******" %>最終是將header.jsp 文件進(jìn)行編譯,然后將其內(nèi)容包含進(jìn)來(lái)。因此include指令包含的是頁(yè)面的內(nèi)容。

  • jsp頁(yè)面的include動(dòng)作元素
    <jsp:include page="/WEB-INF/header.jsp"></jsp:include>
    common.jsp 頁(yè)面的代碼如下:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>common</title>
</head>
<body>
    <h1>This is common page!</h1>
    <jsp:include page="/WEB-INF/header.jsp"></jsp:include>
</body>
</html>

那么按照這種包含的方式,common.jsp 編譯的servlet源碼如下所示:

(common_jsp.java).png

從common_jsp.java的servlet文件中可以看到 <jsp:include page="*****"></jsp:include>,并不是像include指令那樣包含內(nèi)容,而是通過(guò)Tomcat容器中的JspRuntimeLibrary類的include方法進(jìn)行包含。查看Tomcat容器的源碼可以看到,可以看到:

tomcat 源碼.png

從Tomcat源碼可以分析出,<jsp:include page="*****"></jsp:include>的實(shí)質(zhì)是,在common_jsp.java 文件中,執(zhí)行 RequestDispatcher的include方法,而這個(gè)方法使得 common_jsp.java 中的request 對(duì)象、response對(duì)象和header.jsp中的response對(duì)象、request對(duì)象是共享的。

  • RequestDispatcher類的include方法
    request.getRequestDispatcher("/WEB-INF/header.jsp").include(request,response);
    同上分析。

include和共享對(duì)象

對(duì)象 作用
request 頁(yè)面和頁(yè)面構(gòu)成forward以及include關(guān)系,達(dá)成共享
pageContext 在一個(gè)頁(yè)面中共享
session 在整個(gè)會(huì)話期間共享

因此,有上述的分析可以得出這樣的結(jié)論:

  • 對(duì)于 <%@ include file %> 而言,子頁(yè)面只是將內(nèi)容包含在父頁(yè)面之中,因此只有子頁(yè)面和父頁(yè)面之間的request對(duì)象以及pageContext對(duì)象是共享的
  • 對(duì)于 <jsp:include path ></jsp:include>而言,父頁(yè)面通過(guò)RequestDispatcher的include方法將子頁(yè)面包含進(jìn)來(lái),因此只有父頁(yè)面和子頁(yè)面之間request對(duì)象是共享的
include方式 父子頁(yè)面共享對(duì)象
<%@ include file %> pageContext對(duì)象以及request對(duì)象
<jsp:include path ></jsp:include><c:import></c:import> request對(duì)象,不共享pageContext對(duì)象
RequestDispatcher類的include方法 request對(duì)象,不共享pageContext對(duì)象
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容