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程序代碼。當然,代碼與數據的分離的程度還是主要依賴標記庫元素設計的好壞程度。