一、JSP標簽介紹
JSP標簽也稱之為Jsp Action(JSP動作)元素,它用于在Jsp頁面中提供業務邏輯功能,避免在JSP頁面中直接編寫java代碼,造成jsp頁面難以維護。
二、JSP常用標簽
jsp的常用標簽有以下三個
- <jsp:include>標簽
- <jsp:forward>標簽
- <jsp:param>標簽
2.1、<jsp:include>標簽
<jsp:include>標簽用于把另外一個資源的輸出內容插入進當前JSP頁面的輸出內容之中,這種在JSP頁面執行時的引入方式稱之為動態引入。
語法:
<jsp:include page="relativeURL | <%=expression%>" flush="true|false" />
page屬性用于指定被引入資源的相對路徑,它也可以通過執行一個表達式來獲得。
flush屬性指定在插入其他資源的輸出內容時,是否先將當前JSP頁面的已輸出的內容刷新到客戶端。
范例:使用jsp:include標簽引入資源
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> 3 <html> 4 <head> 5 <title>jsp的jsp:include標簽測試</title> 6 </head> 7 8 <body> 9 <%--使用jsp:include標簽引入其它JSP頁面--%> 10 <jsp:include page="/jspfragments/head.jsp"/> 11 <h1>網頁主體內容</h1> 12 <jsp:include page="/jspfragments/foot.jsp"/> 13 </body> 14 </html>
運行結果:
2.2、<jsp:include>標簽與include指令的區別
<jsp:include>標簽是動態引入, <jsp:include>標簽涉及到的2個JSP頁面會被翻譯成2個servlet,這2個servlet的內容在執行時進行合并。
而include指令是靜態引入,涉及到的2個JSP頁面會被翻譯成一個servlet,其內容是在源文件級別進行合并。
通過下面的例子來說明<jsp:include>標簽與include指令的區別
demo.jsp:
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <%! 3 int i=1000; 4 %> 5 <h1>demo.jsp中i的值為:<%=i%></h1>
分別使用include指令和<jsp:include>標簽兩種包含語句,包含以上的demo.jsp
范例:使用@include包含內容
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <%! 3 int i=10; 4 %> 5 <h1>JspIncludeTagDemo01.jsp中i的值為:<%=i%></h1> 6 <h1><%@include file="/jspfragments/demo.jsp"%></h1>
此時在編譯jsp時就已經提示出錯了,如下所示:
這個錯誤說的是變量i已經重復定義了
運行JspIncludeTagDemo01.jsp,結果如下:
運行后發現出現了重復定義變量i的錯誤提示信息,因為靜態包含是將全部內容包含進來之后,再進行處理,屬于先包含后處理。由于被包含進來的頁面demo.jsp中定義了一個變量i,而包含頁面JspIncludeTagDemo01.jsp本身又定義了一個變量i,所以服務器在處理JspIncludeTagDemo01.jsp這個頁面時就會發現里面有兩個重復定義的變量i,因此就會報錯。
而如果現在使用的是<jsp:include>動態包含的話,觀察以下程序:
范例:使用動態包含
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <h1>JspIncludeTagDemo02.jsp</h1> 3 <%! 4 int i=10; 5 %> 6 7 <h1>JspIncludeTagDemo02.jsp中i的值為:<%=i%></h1> 8 <h1><jsp:include page="/jspfragments/demo.jsp" /></h1>
運行結果:
發現結果已經可以正確地顯示,而且不會互相影響,這是因為使用jsp:include屬于動態包含,動態包含就是指先將各個頁面分別處理,處理完之后再將處理后的結果包含進來。
不管是<jsp:include>標簽,還是include指令,它們都會把兩個JSP頁面內容合并輸出,所以這兩個頁面不要出現重復的HTML全局架構標簽,否則輸出給客戶端的內容將會是一個格式混亂的HTML文檔。
2.3、*.jspf擴展名文件在jsp:include、@include和c:import中的區別
JSP規范建議使用.jspf(JSP fragments)作為靜態引入文件的擴展名。今天無意中發現,把一個JSP文件命名為jspf擴展名,然后include到另一個jsp文件中的,發現只有用"@include"指令的時候,jspf文件的內容才會被解析并執行其中的jsp指令和tag,而使用"jsp:include"和JSTL的"c:import"都沒有用,jspf文件被當作純文本文件處理了。
比如現在有一個head.jspf頁面和foot.jspf頁面
head.jspf
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <h1 style="color:red;">網頁頭部</h1>
foot.jspf
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <h1 style="color:blue;">網頁尾部</h1>
首先使用"@include"指令將"head.jspf和foot.jspf" include到IncludeTagTest.jsp頁面,如下所示:
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> 3 <html> 4 <head> 5 <title>jsp的Include指令測試</title> 6 </head> 7 8 <body> 9 <%--使用include標簽引入引入jspf頁面--%> 10 <%@include file="/jspfragments/head.jspf" %> 11 <h1>網頁主體內容</h1> 12 <%@include file="/jspfragments/foot.jspf" %> 13 </body> 14 </html>
運行IncludeTagTest.jsp頁面,運行結果如下:
jspf文件的內容會被解析并執行其中的jsp指令和tag,查看瀏覽器解析JspIncludeTagTest.jsp頁面生成的源代碼,如下所示:
然后再使用<jsp:include>"標簽將"head.jspf和foot.jspf" include到JspIncludeTagTest.jsp頁面中,如下所示:
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> 3 <html> 4 <head> 5 <title>jsp的jsp:include標簽測試</title> 6 </head> 7 8 <body> 9 <%--使用jsp:include標簽引入其它JSPf頁面--%> 10 <jsp:include page="/jspfragments/head.jspf"/> 11 <h1>網頁主體內容</h1> 12 <jsp:include page="/jspfragments/foot.jspf"/> 13 </body> 14 </html>
運行JspIncludeTagTest.jsp頁面,運行結果如下:
查看瀏覽器解析JspIncludeTagTest.jsp頁面生成的源代碼,如下所示:
可以看到,head.jspf和foot.jspf中的<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>沒有解析執行,而是原封不動地作為文本內容輸出到頁面上了,在IE下看不到<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>的輸出,在google和火狐瀏覽器下運行可以看到頁面上輸出<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>,如下所示:
這說明jspf文件Tomcat服務器被當作純文本文件處理了,沒有當作jsp頁面來解析執行,那么該如何解決這個問題呢?如何讓tomcat服務器能夠解析執行*.jspf文件中的java代碼和標簽呢,有如下的幾種解決辦法:
解決辦法一:修改web.xml文件,添加對擴展名為*.jspf文件的映射
如下所示:
1 <!-- 讓jspf擴展名同樣成為JSP Servlet處理的文件。 --> 2 <servlet-mapping> 3 <servlet-name>jsp</servlet-name> 4 <url-pattern>*.jspf</url-pattern> 5 </servlet-mapping> 6 <!-- 讓jsp擴展名同樣成為JSP Servlet處理的文件。 --> 7 <servlet-mapping> 8 <servlet-name>jsp</servlet-name> 9 <url-pattern>*.jsp</url-pattern> 10 </servlet-mapping>
上面的配置方式也可以簡寫成這樣:
1 <servlet-mapping> 2 <servlet-name>jsp</servlet-name> 3 <url-pattern>*.jsp</url-pattern> 4 <!-- 讓jspf擴展名同樣成為JSP Servlet處理的文件。--> 5 <url-pattern>*.jspf</url-pattern> 6 </servlet-mapping>
兩種寫法的效果都是一樣的。
添加這樣的配置信息后,此時tomcat服務器就可以正常解析執行*.jspf文件了,如下所示:
解決辦法二:修改Tomcat服務器的web.xml文件,添加對擴展名為*.jspf文件的映射
找到tomcat服務器的web.xml文件,如下所示:
找到名字為jsp的那個Servlet,如下所示:
1 <servlet> 2 <servlet-name>jsp</servlet-name> 3 <servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class> 4 <init-param> 5 <param-name>fork</param-name> 6 <param-value>false</param-value> 7 </init-param> 8 <init-param> 9 <param-name>xpoweredBy</param-name> 10 <param-value>false</param-value> 11 </init-param> 12 <load-on-startup>3</load-on-startup> 13 </servlet>
然后根據Servlet名找到對應的servlet-mapping配置,如下所示:
1 <!-- The mappings for the JSP servlet --> 2 <servlet-mapping> 3 <servlet-name>jsp</servlet-name> 4 <url-pattern>*.jsp</url-pattern> 5 <url-pattern>*.jspx</url-pattern> 6 </servlet-mapping>
在這里可以看到,名字為jsp的那個Servlet只支持*.jsp和*.jspx兩種擴展名,因此可以在這個地方添加多一個<url-pattern>*.jspf</url-pattern>,如下所示:
1 <!-- The mappings for the JSP servlet --> 2 <servlet-mapping> 3 <servlet-name>jsp</servlet-name> 4 <url-pattern>*.jsp</url-pattern> 5 <url-pattern>*.jspx</url-pattern> 6 <url-pattern>*.jspf</url-pattern> 7 </servlet-mapping>
經過這樣的配置之后,Tomcat服務器就可以正常解析和執行*.jspf文件了。
2.3、<jsp:forward>標簽
<jsp:forward>標簽用于把請求轉發給另外一個資源。
語法:
<jsp:forward page="relativeURL | <%=expression%>" />
page屬性用于指定請求轉發到的資源的相對路徑,它也可以通過執行一個表達式來獲得。
范例:使用<jsp:forward>標簽跳轉頁面
forwarddemo01.jsp
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <%--使用<jsp:forward>標簽跳轉到forwarddemo02.jsp--%> 3 <jsp:forward page="/forwarddemo02.jsp"/>
forwarddemo02.jsp
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <h1>跳轉之后的頁面!!</h1>
運行結果如下:
從頁面的顯示效果來看,頁面已經完成了跳轉,但地址欄沒有變化,地址欄中顯示的地址還是forwarddemo01.jsp,但頁面顯示的內容卻是forwardemo02.jsp中的內容。因為此跳轉屬于服務器端跳轉。只要是服務器端跳轉,則地址欄永遠沒有變化。
2.4、<jsp:param>標簽
當使用<jsp:include>和<jsp:forward>標簽引入或將請求轉發給其它資源時,可以使用<jsp:param>標簽向這個資源傳遞參數。
語法1:
<jsp:include page="relativeURL | <%=expression%>">
<jsp:param name="parameterName" value="parameterValue|<%= expression %>" />
</jsp:include>
語法2:
<jsp:forward page="relativeURL | <%=expression%>">
<jsp:param name="parameterName" value="parameterValue|<%= expression %>" />
</jsp:include>
<jsp:param>標簽的name屬性用于指定參數名,value屬性用于指定參數值。在<jsp:include>和<jsp:forward>標簽中可以使用多個<jsp:param>標簽來傳遞多個參數。
范例:使用<jsp:param>標簽向被包含的頁面傳遞參數
JspIncludeTagDemo03.jsp
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <h1>JspIncludeTagDemo03.jsp</h1> 3 <hr/> 4 <jsp:include page="/jspfragments/Inc.jsp"> 5 <jsp:param name="parm1" value="hello" /> 6 <jsp:param name="parm2" value="gacl" /> 7 </jsp:include>
Inc.jsp
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <h1>接收從JspIncludeTagDemo03.jsp頁面中傳遞過來的參數:</h1> 3 <h2><%=request.getParameter("parm1")%></h2> 4 <h2><%=request.getParameter("parm2")%></h2>
在JspIncludeTagDemo03.jsp頁面中使用<jsp:include>標簽將Inc.jsp頁面包含進來,使用<jsp:param/>標簽向Inc.jsp頁面傳遞了兩個參數parm1和parm2
JspIncludeTagDemo03.jsp頁面運行結果如下:
范例:使用<jsp:param>標簽向要跳轉的頁面傳遞參數
forwarddemo03.jsp
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <%--使用<jsp:forward>標簽跳轉到forwarddemo04.jsp--%> 3 <%--使用<jsp:param>標簽向forwarddemo04.jsp傳遞參數--%> 4 <jsp:forward page="/forwarddemo04.jsp"> 5 <jsp:param name="ref1" value="hello" /> 6 <jsp:param name="ref2" value="gacl" /> 7 </jsp:forward>
forwarddemo04.jsp
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> 2 <h1>跳轉之后的頁面!!</h1> 3 <h1>接收從forwarddemo03.jsp傳遞過來的參數:</h1> 4 <h1>ref1:<%=request.getParameter("ref1")%></h1> 5 <h1>ref2:<%=request.getParameter("ref2")%></h1>
運行結果如下:
文章列表