本文是 JSTL 系列的第二篇-- JSP 標準標簽庫.相關資源:JSP Standard Tag Libraries, Part 1(http://www.onjava.com/pub/a/onjava/2002/03/13/jsp.html)
??
?? 此文中,我們將更詳細地講述不同標記描述文件(Tag Library Descriptor,TLD)里的各種標記。我們會通過例子去說明條件、迭代、URL、SQL及XML等標記的使用。文章的目的就是展示JSTL的主要功能,說明如何使用JSTL以及說明如何使用JSTL去改善JSP的開發。
?? 簡單來說,JSTL就是一套可以用在多個方面的自定義功能,JAVA標準組織(Java Community Process,JCP)在JSR-52里面定義到,JSTL提供表達式語言(expression language),流程控制及數據檢校等功能,JSTL規范的最后版本可以在JCP的網站上找得到。
?? JSTL要求運行在支持JSP 1.2版的服務器上,JSTL的主旨就是簡化JSP的開發以及提供訪問、操作數據的簡便方式,使用JSTL可以使用大家的工作更輕松。
????JSTL包含了多個TLD和JAR文件,這些TLD覆蓋很多功能,我們將一個個說明。首先,我們要重點說一下JSTL最突出的功能,表達式語言。
????實際上,表達式語言是由JSR-152專家組所提供的,也就是定義JSP 2.0的專家組。JSR-152和JSR-52很多時是在一起工作的,更甚的是,表達式語言已經成為JSP 2.0的一部分。表達式語言提供訪問數據的一種簡便方法,它支持運算,支持bean,支持集合,同時也支持類型轉換和屬性的缺省值。
使用表達式語言?? 表達式語言是通過 ${expression} 這樣的形式來調用的,并且它只能用在屬性里面,例如:
<c:if test="${product.price >= customer.limit}">
...
</c:if>
?? 在上面的例子,我們使用表達式語言做一個比較運算,它也可以混合一些靜態文本,就象下面的例子:
<c:forEach var="current">
????????<c:out value="Product-${current}"/>
</c:forEach>
?? 在例子里面,我們枚舉出一個集合里面的元素,之后將它加在一個字符串后面輸出,結果就象:
Product-1
Product-2
Product-3...
?? 從例子里可以看出,相對于以前的代碼,使用JSTL之后,在整個過程變得簡單清晰。在以前,你需要定義一個對象,要知道對象的類型,并且要掌握一些腳本語言,而這只是去完成一個簡單的操作。
?? 現在,有了JSTL,我們可以使用簡單的語法去訪問數據,表達式語言是非常適合做這樣的工作的。
?? 再舉一個例子,我們可以將:
<jsp:useBean id="customer" type="sample.Customer" scope="request"/> ...
Customer Name: <%=customer.getName()%>
...
<% if (customer.getState().equals("CO")){ %>
...
<%}%>
?? 轉換成:
Customer Name: ${ customer. name}
<c:if test="${customer. state == param. state}">
...
</c:if>
?? 表達式語言允許直接訪問JSP中任何作用域的變量,比如可以用${foo}來代替pageContext.findAttributes("foo")。還有可以能過點或索引來訪問bean以及它們的屬性,如:
${user.address.city}
${products.[product.id]}
${products.["Part-010"]}
????表達式語言提供所有你使用到的運算:==, !=, <,>, <=, >=, &&, ||, ! 。這表達式語言中,它們表示為lt, gt, le, ge, eq, ne,這樣可以避免與XML的語法重復。表達式語言也有普通的算術運算同布爾運算。另外,表達式語言還具有自動類型轉換的功能,如int value = "${request.myValue}"會自動轉換。
?? 在表達式語言是可以提供屬性的缺省值的,使用缺省值是為了避免空指針異常,下面是一個例子,
<c:set var="city" value="${user.address.city}" default="N/A" />
?? 現在,我們已經對表達式語言有所了解,下面,讓我們看一下與EL相關的操作。
核心操作?? 在核心標記庫里使用到表達式語言。<c:out>標記將表達式語言所計算出的值輸出到當前的JSP輸出流。這個與JSP中的<%= scripting exp %>相似。比如:
<c:out value="${customer.name}"??default="N/A" />
?? 核心標記庫里還可以設置和清除變量。變量的缺省作用域是PAGE。例如我們可以用<c:set var="customer" value=${customer}" />設置一個頁面變量customer,再用一個<c:remove var="customer" />去清除它。
?? 現在,能過JSTL,我們可以使用一個標記去捕捉java.lang.Throwable,例如:<c:catch var="myError" />。使用這個標記,可以統一頁面的異常處理,但這不意味著代替JSP的錯誤頁面機制。通過使用<c:catch>標記,就可以在頁面上控制特定的異常,而不用轉到錯誤頁面,其實并不是所有異常都需要轉到錯誤頁面。通過使用<c:catch>標記,與用戶的交互也變得更友好。
條件操作????在條件操作中使用EL是簡化JSP的一種強有力的手段。<c:if>標記,它可以構造簡單的條件表達式。下面的例子,訪問了一個對象的屬性:
<c:if test="${user.visitCount == 1}"
????????Welcome back!
</c:if>
????當然,有IF就一定會有ELSE。如果是"if/then/else"結構的,則使用<c:choose>, <c:when>和<c:otherwise>標記。
?? 看一看下面的例子,我們對一個查詢結果集做過了一些處理,之后使用標記去顯示正確的信息。
<c:choose>
<c:when test="${count == 0}">
????????No records matched your selection.
</c:when>
<c:otherwise>
????????<c:out value="${count}"/> records matched your selection.
</c:otherwise>
</c:choose>
迭代操作?? 在整個JSTL里面,可能最有用的操作就是迭代操作了。迭代操作的標記有<c:forEach>,<c:forToken>和<x:forEach>。最后一個是XML的迭代操作。下面我們會說到XML的操作,但現在我們還是繼續核心操作。
????迭代操作支持所有的J2SE集合類型,包括List, LinkedList, ArrayList, Vector, Stack, 和Set。還有java.util.Map對象,如HashMap, Hashtable, Properties, Provider, 和Attributes。還有數組。當使用基本數據類型的數組時,它會包裝成對應的基本類。在迭代操作中,會輸出兩個東西,當前的數據項和迭代的狀態??纯聪旅娴睦樱?br />
<table>
<c:forEach var="product"
??????????????items="${products}"
??????????????varStatus="status">
<tr>
????????<td><c:out value="${status.count}"/></td>
????????<td><c:out value="${product.name}"/></td>
</tr>
</c:forEach>
</table>
?? 例子中,products是一個集合,當前的數據項入在變量product里面,變量status是當前的迭代狀態。是不是很簡單。
URL操作?? 除了迭找操作外,核心標記庫里也提供了URL相關的操作,它包括超鏈接,引入和重定向。可以使用<c:url>去設定一個URL。假如我們想指定一個帶參數的URL,并在鏈接中使用,那就象下面的例子:
<c:url=http://mysite.com/register var="myUrl">
????????<c:param name="name" value="${param.name}"/>
</c:url>
<a href='<c:out value="${myUrl}"/>'>Register</a>
?? JSTL中有很強大的資源引入機制,它可以指定絕對的URL,在同一應用的相對的URL,不同應用的相對的URL,還有FTP資源。下面給出一些例子:
絕對URL:<c:import url="http://sample.com/Welcome.html"/>
同一應用中的相對的URL:<c:import url="/copyright.html"/>
不同應用中的相對的URL:<c:import url="/myLogo.html" context="/common"/>
FTP資源:<c:import url="ftp://ftp.sample.com/myFile"/>
?? 通過上面的說明可以看出,<c:import>比<jsp:include>強大得多,但是,也有其它理由去使用<jsp:include>的。在JSTL中,對于引入的資源使用了緩存,但有時候緩存會是沒用的。如果你使用<jsp:include>,資源的內容會被讀入并寫到當前的JspWriter,而且是每訪問一次就重讀一次。而使用<c:import>,資源的內容則只會讀取一次。
本地化操作?? JSTL的另一樣重要功能就是本地化操作,通過當前請求的,或者環境配置中的參數,就可以簡便地實現本地化操作。這個操作是使用J2SE中的ResourceBundle機制去存放各種譯文的。JSTL通過設定的區域,去找到并使用相應的ResourceBundle。用<fmt:setLocale>去設置區域,如<fmt:setLocale value="es_Es"/>,value屬性為語言代碼和國家代碼。也可以直接指定一個ResourceBundle:<fmt:bundle basename="ApplicationResource_fr"/>。
?? 一旦設置了區域或者綁定特定的信息,那么<fmt:message>標記就會自動地選擇正確的ResourceBundle,使用以下的形式就可以輸出正確的信息:
<fmt:message key="welcome">
????????<fmt:param value="${visitCount}" />
<fmt:message/>
?? 你也可以直接使用< fmt:requestEncoding/>標記去設置當前請求的字符集。
?? 獲取和顯示文字信息只是本地化操作的一半,而另一半就是格式化同解釋日期和數字,不同的地域會有不同的日期和數字格式的。使用<fmt:formatNumber>和<fmt:parseNumber>去格式化數字,金額,百分比數,而且還可以指定格式,就如<fmt:formatNumber value="12.3" pattern=".00"/>會輸出"12.30"。
?? 日期和時間的處理使用<fmt:formatDate>, <fmt:timeZone>, <fmt:setTimeZone>, 和 <fmt:parseDate>。
SQL操作?? SQL操作允許你直接操作數據源,在MVC模式里面是不提倡這樣做的,我更是徹底反對在一個正式產品里這樣做。它們只適用于快速開發,原型開發或者是一些小的應用里面,它們是不應該用在一些大型的應用上面的。不過也有許多開發人員想使用它,所以它保留在標準里面。下面我們來看一下SQL操作。
?? SQL標記可以用來設置數據源,執行查詢,訪問查詢結果,執行更新等。所有的SQL操作都是基于某一數據源的。有幾種方式可以設置數據源:在配置文件里設置
sql.datasource參數,在程序中直接設置,或者使用<sql:setDataSource>標記。如下面就設置了一個MySql的數據源:
<sql:setDataSource var="datasource"??driver="org.gjt.mm.mysql.driver" url="jdbc:mysql://localhost/db" />
?? 與JDBC中的DriverManager相似,<sql:setDataSource>也只是一個包裝。數據源的屬性,它可以是一個JNDI資源,也可以是一個JDBC參數。用<sql:query datasource="${datasource}" ... />這樣的方式來訪問數據。
?? 我們把這些東西都放在一起,它是會是這樣的:
<sql:query var="customer" datasource="${datasource}"
SELECT * FROM customers WHERE state = 'CO' ORDER BY city
</sql:query>
<table>
<c:forEach var="row" items="${customers.row}">
????????<tr>
????????<td><c:out value="${row.custName}" /></td>
????????<td><c:out value="${row.address}" /></td>
????????</tr>
</c:forEach>
</table>
????使用事務和執行更新一樣也很簡單。例如,我們建立一個事務,并執行幾個更新,那就是下面代碼:
<sql:transaction dataSource="${dataSource}">
<sql:update>
????????UPDATE account SET Balance =Balance -? WHERE accountNo = ?
????????<sql:param value="${transferAmount}"/>
????????<sql:param value="${accountFrom}"/>
</sql:update>
</sql:transaction>
?? 用<sql:dateParam>標記來設定SQL語句中的類型為日期型的參數的值。
XML操作?? JSTL中最后一類操作是XML操作,XML操作也可細分成核心操作,流程控制操作和轉換操作。JSTL中的XML操作是基于Xpath的,Xpath是XML操作專用的表達式語言。JSTL所有的XML操作中,用"select"屬性去指定XPath表達式,這些信息將由XPath引擎解釋。
?? XML核心操作與JSTL的核心操作相似,它包含<x:out>, <x:set>, 和<x:parse>標記。<x:parse>標記提供將XML文檔轉換成結構化數據的功能,之后,這些數據就能被XPath引擎解釋了。例如,有一個關于書籍的XML文檔,我們就可以解釋它,并打印出來:
<c:import url="http://oreilly.com/book?id=1234" var="xml"/>
<x:parse source="${xml}" var="bookInfo"/>
<x:out select="$bookInfo/title"/>
<x:out select="$bookInfo/author"/>
?? XML類流程控制操作與核心類的流程控制操作一樣,包括if, choose, when, otherwise, 和 forEach 這些標記,區別只是它們使用XPath表達式語言。當表達式計算出來后,根據一些規則,結果將轉換成一個布爾值。這些規則如下:
A number is true if and only if it is neither positive or negative zero nor NaN.
一個數字當且僅當它不是一個正數,也不是一個負零,更不是一個NaN(非數字)時,表達式的值為真。
A node-set is true if and only if it is non-empty.
一個節點當且僅當它不是空節點時,表達式的值為真。
A string is true if and only if its length is non-zero.
一個字符串當且僅當它的長度不為0是,表達式的值為真。
?? XML轉換操作,是通過XSL樣式表來轉換XML文檔,轉換的結果輸出到當前頁面,另一方面也可以其中的結果保存到變量中。完成一個轉換就是導入一個XML文檔和XSL樣式表,之后做一個轉換那么簡單:
<c:import url="/books" var="xml"/>
<c:import url="/WEB-INF/xslt/bookDisplay.xsl" var="xslt"/>
<x:transform source="${xml}" xslt="${xslt}"/>
?? 如果你在轉換中使用到參數,你可以用<x:param>指定參數名稱及參數值。
小結?? 現在,你已經學會了使用JSTL中的標記,這樣,開發JSP會更容易更快捷。留意JSTL的發展,當JSTL的正式版本發布以后,JSP服務器的提供廠商就會對此優化。你可以在http://jakarta.apache.org/taglibs/doc/standard-doc/intro.html中得到JSTL的最后實現版本。