天極網5月2號發表的《JSP與XML的結合》一文中對JSP與XML的互動只提到一種使用標記庫的方法,JSP與XML互動還有兩種重要的方法,因此在本文中我想簡要闡述一下,以作為《JSP與XML的結合》的補充。
使用JavaServer Pages有三種不同的方法用於處理XML文檔,每種方法都有利於提高分離頁面代碼與XML資料的水平,有利於簡化開發網頁的複雜度並且改善提高組件與頁面代碼的可重用性。
JavaServer Pages ( JSP)和XML是Sun的J2EE的兩個至關緊要的組件。 JSP是用於建立應用程式伺服器端程式的有效工具,而客戶可以是一個瀏覽器,一個裝置或其它的應用程式。 你可以使用XML描述資料並在聯絡伺服器與系統的其餘部分之間傳遞。 如果你仔細考慮Web服務的抽象概念的話,JSP可以被認為是實現技術而XML則是資料封裝和訊息傳送技術。 JSP頁面可以通過三種方式使用XML:直接使用XML檔案,使用JavaBeans來執行XML處理或者通過標記庫使用XML。
一、直接使用XML
我們可以在JSP頁面中直接使用XML,這要分為三類:
1. JSP可以讀取XML檔案並且基於這些資料執行動作。例如:一應用程式可以讀取具有某些特定結構的資料的XML檔案。
2. JSP可以建立XML檔案以發送資料到客戶程式或其它的應用程式。 JSP可以轉換XML檔案,這個變換可以是交給XSLT處理,由JSP作為控制器,或者通過非XSLT解決方案來完成。 在這兩種情況下,JSP的作用都是讀取XML檔案,轉換它並產生一個輸出。
因為JSP包含內嵌的Java程式,所以它可以直接地調用一個剖析器來讀/寫xml資料。 這是非常不合理的方法,因為資料和代碼邏輯並不能很好的被分離開。 另外,這樣的程式也很難讀得懂。 所以,下面我將介紹第二方法:使用JavaBeans。
二、使用JavaBeans
JSP可以通過<jsp:usebean>標記與JavaBeans緊密的整合起來。 下面的程式片段示範了如何使用在JSP頁面中的一個JavaBean來設定並擷取屬性。
<jsp:useBean id="cb" scope="session" class="xmlrep.Customer" /> <jsp:setProperty name="cb" property="id" value="45" /> <B> First Name is: </B> <%=cb.getFname() %> <p> <B> Last Name is: .</B> <%= cb.getLname() %> |
JSP與JavaBeans整合的特徵就是可以自動把超文本標誌語言的表單元素翻譯成JavaBean屬性。 如果有一個HTML表單並且想讓它向JavaBean提交表單內容,你可以寫下面的代碼:
<jsp:setProperty name="cb" property="*" /> |
name屬性包含JSP頁面已經引用Bean的值。 前面<jsp:useBean>標記設定名稱為" cb "。 與設定單獨的Bean屬性不同,你可以使用星號來標誌"全部的"屬性。 JSP頁面自動地映射HTML表單值為同名的Bean屬性。 如果你讀取每個HTML表單元素,然後調用相應屬性的Bean設定方法,那麼結果也將是相同的。
可以看到,類XML的標記允許JSP頁面訪問JavaBeans。 我們通過儘可能多的把封裝的代碼轉變成可重用組件(JavaBeans),就可以把JSP頁面中代碼最佳化到最小的程度。 你可以使用通用的文法剖析器,像Xerces或者JAXPI,在單獨的JavaBeans裡與XML檔案互動作用--並且你可以在不改動JSP頁面的情況下改變剖析器。 此外,Beans還可以使用XSLT來執行XML檔案的轉換。
使用JSP和JavaBeans來完成的這些抽象的動作比直接在JSP頁面中插入原始的Java程式要好得多,但是你仍然需要熟悉Java程式,以便隨時改變JSP頁面。 應用程式的一致性與條理性依靠JavaBeans合作建立一個統一的輸出結果的好壞程度。 例如,Bean中的缺陷可能會造成整個XML輸出無效。 依靠Beans指定資源的方法,可能同時帶來執行效能的問題。
三、通過標記庫JSP與XML互動
這也就是在前文中重點提到的,但是因為它實在太重要了,所以在本文中我必須也要提到。標記庫可以定義出現在JSP頁面中的作為類XML元素的自訂標籤,可以把特定的Java代碼與每個標記都關聯起來。 例如,假定你可以訪問一個天氣情況資料庫,而且你需要輸出現在的天氣狀況。 那麼,你可以在JSP中插入JDBC程式碼直接查詢資料庫(雖然這並不是一個好的選擇),把這些代碼封裝到一個JavaBean內,或者把它封裝成一個標記庫。 使用最後一個選擇,你的JSP頁面內的程式碼看上去就像:
<%@taglib uri="the TLD file" prefix="foo" %> Current weather is <foo:Weather/> |
注意在上述程式碼中看不到任何Java代碼的蹤跡。 作為一個頁面設計人員,你使用一種熟悉的文法就像<foo:weather/>,這些與其它的任何標記看起來非常相似。 在頁面中包含現在的天氣狀況的HTML字串的地方插入它。
標記庫有一個關聯的XML格式的描述符檔案,名叫Tag Library Descriptor(標記庫描述符,TLD)。 在TLD檔案中的每個標記都有一個條目,包括它的名稱,版本和其它的任選資訊。 在JSP頁面內,你可以用" <%@_taglib prefix = " foo " %> "指令指定TLD檔案。 屬性" prefix "用來指定一個首碼,用來在JSP頁面內使用特定的庫來引用任何標記。 那為什麼我們要使用標記<foo:Weather/>而不僅僅是<Weather/>。 TLD檔案的精確位置依賴於正在使用的應用程式伺服器。
一個標記庫標記可以代替為了完成這段程式邏輯的相應的Java程式碼。 每個標記都相當於一個相同名稱的Java類。 這個類必須實現TagSupport介面,包含捕獲事件觸發程序方法作為處理這個頁面的JSP引擎。 當它第一次遇到這個標記的時候,引擎就會調用doStartTag ()方法。 可以使這個方法為空白或者當需要的時候才執行應用程式邏輯。 當該方法返回SKIP_BODY時,那麼引擎跳過這個標記體。 當它返回EVAL_BODY_INCLUDE時,引擎將會處理這個標記以及它的子標記。 類似地,JSP引擎在分析了結束標記之後就會調用doEndTag ()方法。 doAfterBody ()方法讓你可以在引擎處理元素體之後執行動作,但是必須在doEndTag ()方法作用之前。 下面就是實現天氣狀況的Weather類的一個樣本程式碼:
import javax.servlet.jsp.*; import javax.servlet.jsp.tagext.*; import java.io.*; public class Weather extends TagSupport { public int doStartTag() { try { JspWriter out = pageContext.getOut(); out.print(" sunny and cloudy mixed with " + "rain and sunshine. "); } catch (IOException e) { System.out.println("Error " + e); } return (SKIP_BODY); } } |
當引擎遇到" <somePrefix:Weather/> "標記的時候,它會在標記庫內搜尋一個同名的類。 如果doStartTag ()方法被實現的話(在本例中是這樣的),它就會被調用。 這就使字串包含適應顯示的天氣情況。 因為方法返回了SKIP_BODY,JSP讀取器移動到標記的末尾。
最簡單的使用JSP和標記庫的方法就是使用Apache Tomcat引擎。 這個引擎也充當Servlet和JSP應用程式介面的引用實現。
當使用標記庫時,JSP頁面看上去就非常像XML檔案了。 當JSP頁面被處理時,引擎執行與標記相關聯的程式碼(實際上,首先調用JSP引擎把JSP頁面翻譯成一個servlet,然後再編譯servlet。 與標記庫相關聯的方法都被包含在servlet中。),一個熟悉XML的人就可以設計並使用各種各樣的頁面配置進行實驗,而不必改變任何Java程式碼。當然,代碼與資料的分離的程度還是主要依賴標記庫元素設計的好壞程度。