茫茫網海中的冷日
         
茫茫網海中的冷日
發生過的事,不可能遺忘,只是想不起來而已!
 恭喜您是本站第 1675477 位訪客!  登入  | 註冊
主選單

Google 自訂搜尋

Goole 廣告

隨機相片
IMG_2335131.jpg

授權條款

使用者登入
使用者名稱:

密碼:


忘了密碼?

現在就註冊!

爪哇咖啡屋 : [分享]Struts,MVC的一種開放原始碼實現

發表者 討論內容
冷日
(冷日)
Webmaster
  • 註冊日: 2008/2/19
  • 來自:
  • 發表數: 15771
[分享]Struts,MVC的一種開放原始碼實現
本文介紹 Struts,它是使用 servlet 和 JavaServer Pages技術的一種 Model-View-Controller 實現。Struts 可幫助您控制 Web項目中的變化並提高專業化水平。儘管您可能永遠不會用 Struts實現一個系統,但您可以將其中的一些思想用於您以後的 servlet 和 JSP網頁的實現中。

簡介
小學生也可以在因特網上發布 HTML網頁。但是,小學生的網頁和專業開發的網站有質的區別。網頁設計人員(或者HTML開發人員)必須理解顏色、用戶、生產流程、網頁布局、瀏覽器兼容性、圖像創建和JavaScript 等等。設計漂亮的網站需要做大量的工作,大多數 Java開發人員更注重創建優美的對象接口,而不是用戶界面。JavaServer Pages(JSP) 技術為網頁設計人員和 Java 開發人員提供了一種聯繫鈕帶。

如果您開發過大型 Web應用程序,您就理解 變化 這個詞的含義。「模型-視圖-控制器」(MVC)就是用來幫助您控制變化的一種設計模式。MVC減弱了業務邏輯接口和數據接口之間的耦合。Struts 是一種 MVC實現,它將 Servlet 2.2 和 JSP 1.1 標記(屬於 J2EE規範)用作實現的一部分。儘管您可能永遠不會用 Struts實現一個系統,但了解一下 Struts或許使您能將其中的一些思想用於您以後的 Servlet 的 JSP 實現中。

在本文中,我將以一個 JSP文件為起點討論該網頁的優缺點,該文件中使用的元素可能是您所熟悉的。隨後我將討論Struts,並說明它是如何控制您的 Web項目中的變化並提高專業化水平的。最後,我將重新開發這個簡單的 JSP文件,在開發過程中我已顧及到網頁設計人員和變化。

一個 JSP 文件就是一個 Java servlet

JavaServer Page (JSP) 文件只是審視 servlet 的另一種方式。JSP文件的概念使我們能夠將 Java servlet 看作一個 HTML 網頁。JSP 消除了Java 代碼中經常出現的討厭的 print() 語句。JSP文件首先被預處理為 .java 文件,然後再編譯為.class 文件。如果您使用的是 Tomcat,則可以在work 目錄下查看預處理後的 .java文件。別的容器可能將 .java 和 .class文件存儲在其他位置;這個位置與容器有關。圖 1 說明了從 JSP 文件到servlet 的流程。

圖 1. 從 JSP 文件到 servlet的流程
從 JSP 到 servlet 的流程

(這與 Microsoft 的 Active Server Page (ASP) 明顯不同。ASP被編譯到內存中,而不是編譯到一個單獨的文件中。)

簡單的獨立 JSP 文件
在小型 JSP應用程序中,經常會看到數據、業務邏輯和用戶界面被組合在一個代碼模塊中。此外,應用程序通常還包含用來控制應用程序流程的邏輯。清單1 和圖 2 展示了允許用戶加入一個郵件列表的一個簡單 JSP 文件。

清單 1. join.jsp -- 一個簡單的請求和響應 JSP文件

<%@ page language="java" %>
  <%@ page import="business.util.Validation" %>
    <%@ page import="business.db.MailingList" %>
      <% String error = ""; String email = request.getParameter("email"); // 是否有電子郵件地�} 
        if( email!=null ) { // 驗證輸入... 
        if( business.util.Validation.isValidEmail(email) ) { // 存儲輸入... 
        try { business.db.MailingList.AddEmail(email); } 
        catch (Exception e) 
        { error = "Error adding email address to system." + e; } 
        if( error.length()==0 ) { %> // 重定向到歡迎頁...
        <jsp:forward page="welcome.html"/>
        <% 
        } 
        } else { // �]置錯誤消息並重新顯示網頁 
        error = email + " is not a valid email address, please try again."; } } 
        else { email = ""; } %>
          <html>
            <head>
              <title>
                Join Mailing List
              </title>
            </head>
            <body>
              <font color=red>
                <%=error%>
              </font>
              <br>
              <h3>Enter your email to join the group</h3>
              <form action="join.jsp" name="joinForm">
                <input name="email" id="email" value=<%=email%>
                >
                </input>
                <input type=submit value="submit">
              </form>
            </body>
          </html> 

圖 2. 在簡單的請求和響應中,JSP文件設置數據、控制到下一個網頁的流程並創建 HTML
簡單的請求和響應 JSP

這個郵件列表 JSP文件是一個獨立的、自主完成所有任務的模塊。未包含在這個 JSP文件中的僅有代碼是包含在 isValidEmail()中的實際驗證代碼和將電子郵件地址存入數據庫的代碼。(將isValidEmail()方法分離到可重用的代碼中似乎是當然的選擇,但我曾見過直接嵌入網頁中的isValidEmail()代碼。單頁方法的優點是易於理解,並且最初也易於構建。此外,對於各種圖形化開發工具,入門也很容易。

join.jsp 的活動

  1. 顯示打開的輸入網頁。
  2. 從表單參數中讀取 email 的值。
  3. 驗證 email 地址。
  4. 如果 email 地址有效:
    • 將該地址添加到數據庫中。
    • 重定向到下一個網頁。
  5. 如果 email 地址無效:
    • 設置錯誤消息。
    • 重新顯示含有錯誤消息的 join.jsp。

單頁方法的後果

  • HTML 和 Java 強耦合在一起
    JSP 文件的編寫者必須既是網頁設計者,又是 Java開發者。其結果通常要麼是很糟的 Java代碼,要麼是難看的網頁,有時甚至 Java 代碼和網頁都很糟。

  • Java 和 JavaScript 的不足
    隨著網頁逐漸變大,很容易想到實現一些 JavaScript。當網頁中出現JavaScript 時,這種腳本就可能與 Java代碼產生混淆。可能產生混淆的一個例子是使用客戶端的 JavaScript來驗證 email 域。


  • 內嵌的流程邏輯
    要理解應用程序的整個流程,您必須瀏覽所有網頁。試想一下擁有 100個網頁的網站的錯綜複雜的邏輯。

  • 調試困難
    除了很糟的外觀之外,HTML 標記、Java 代碼和 JavaScript代碼都集中在一個網頁中還使調試變得相當困難。

  • 強耦合
    更改業務邏輯或數據可能牽涉相關的每個網頁。

  • 美學
    在很大的網頁中,這編碼樣式看起來雜亂無章。我過去進行 Microsoft ASP開發時,我經常看到有 1000行的網頁。即使有彩色語法顯示,閱讀和理解這些代碼仍然比較困難。

請別在我的 HTML 中加入太多的 Java代碼
在清單 1 中,不是 Java 代碼中有大量的 HTML,而是在 HTML文件中有大量的 Java 代碼。從這個觀點來看,除了允許網頁設計人員編寫Java 代碼之外,我實際上沒做什麼。但是,我們並不是一無所有;在 JSP1.1 中,我們獲得一種稱為「標記」的新特性。

JSP 標記只是將代碼從 JSP 文件中抽取出來的一種方式。有人將 JSP標記看作是 JSP 文件的宏,其中用於這個標記的代碼包含在 servlet中。(宏的觀點在很大程度上是正確的。)出於同樣的原因,我不希望在Java 代碼中看到 HTML 標記,我也不希望在 JSP 文件中看到 Java代碼。JSP 技術的整個出發點就是允許網頁設計人員創建servlet,而不必糾纏於 Java 代碼。標記允許 Java 程序員將 Java代碼偽裝成 HTML 來擴展 JSP 文件。圖 3 顯示了從 JSP網頁中抽取代碼並將它們放入 JSP 標記中的一般概念。

圖 3. JSP 標記

JSP 標記分解

清單 2 是用來說明 Struts 標記的功能的一個例子。在清單 2中,正常的 HTML<form> 標記被用 Struts<form:form> 標記替換。清單 3顯示了瀏覽器接收到的結果 HTML。瀏覽器獲得 HTML<form> 標記,但帶有附加代碼,如JavaScript。附加的 JavaScript 激活 email地址域。服務器端的<form:form> 標記代碼創建適當的 HTML,並使網頁設計人員不再接觸 JavaScript。

清單 2. Struts 的 form 標記

          <form:form action="join.do" focus="email" >
            <form:text property="email" size="30" maxlength="30"/>
            <form:submit property="submit" value="Submit"/>
            </form:form> 


清單 3. 發送給瀏覽器的結果 HTML


            <form name="joinForm" method="POST" action="join.do;jsessionid=ndj71hjo01">
              <input type="text" name="email" maxlength="30" size="30" value="">
              <input type="submit" name="submit" value="Submit">
            </form>
<script language="JavaScript">
<!--
document.joinForm.email.focus()
// -->
</script> 

有關 JSP 標記的注意事項:

  • JSP 標記需要一個運行 JSP 1.1 或更高版本的容器。

  • JSP 標記在服務器上運行,而不像 HTML標記那樣由客戶機解釋。

  • JSP 標記提供了適當的代碼重用機制。

  • 可以使用一種稱為 include 的 JSP 機制將 HTML 和JavaScript 添加到網頁中。但是,開發人員常常會創建巨大的 JavaScript庫文件,這些庫文件被包含在 JSP 文件中。結果返回給客戶機的 HTML網頁要比必需的 HMTL 網頁大得多。include的正確用法是僅將它用於生成諸如頁眉和頁腳這類內容的 HTML代碼段。

  • 通過抽取出 Java 代碼,JSP 標記使開發角色更加專業化。

模型-視圖-控制器 (MVC)
JSP標記只解決了部分問題。我們還得處理驗證、流程控制和更新應用程序的狀態等問題。這正是MVC 發揮作用的地方。MVC通過將問題分為三個類別來幫助解決單一模塊方法所遇到的某些問題:

  • Model(模型)
    模型包含應用程序的核心功能。模型封裝了應用程序的狀態。有時它包含的唯一功能就是狀態。它對視圖或控制器一無所知。

  • View(視圖)
    視圖提供模型的表示。它是應用程序的 外觀 。視圖可以訪問模型的讀方法,但不能訪問寫方法。此外,它對控制器一無所知。當更改模型時,視圖應得到通知。

  • Controller(控制器)
    控制器對用戶的輸入作出反應。它創建並設置模型。

MVC Model 2

Web向軟件開發人員提出了一些特有的挑戰,最明顯的就是客戶機和服務器的無狀態連接。這種無狀態行為使得模型很難將更改通知視圖。在Web上,為了發現對應用程序狀態的修改,瀏覽器必須重新查詢服務器。

另一個重大變化是實現視圖所用的技術與實現模型或控制器的技術不同。當然,我們可以使用Java(或者 PERL、C/C++ 或別的語言)代碼生成HTML。這種方法有幾個缺點:

  • Java 程序員應該開發服務,而不是 HTML。
  • 更改布局時需要更改代碼。
  • 服務的用戶應該能夠創建網頁來滿足它們的特定需要。
  • 網頁設計人員不能直接參與網頁開發。
  • 嵌在代碼中的 HTML 很難看。

對於 Web,需要修改標準的 MVC 形式。圖 4 顯示了 MVC 的 Web改寫版,通常也稱為 MVC Model 2 或 MVC 2。

圖 4. MVC Model2
MVC Model 2

Struts,MVC 2 的一種實現
Struts 是一組相互協作的類、servlet 和 JSP標記,它們組成一個可重用的 MVC 2 設計。這個定義表示 Struts是一個框架,而不是一個庫,但 Struts也包含了豐富的標記庫和獨立於該框架工作的實用程序類。圖 5 顯示了Struts 的一個概覽。

圖 5. Struts概覽

Struts 概覽

Struts 概覽

  • Client browser(客戶瀏覽器)
    來自客戶瀏覽器的每個 HTTP 請求創建一個事件。Web 容器將用一個 HTTP響應作出響應。

  • Controller(控制器)
    控制器接收來自瀏覽器的請求,並決定將這個請求發往何處。就 Struts而言,控制器是以 servlet實現的一個命令設計模式。struts-config.xml文件配置控制器。

  • 業務邏輯
    業務邏輯更新模型的狀態,並幫助控制應用程序的流程。就 Struts而言,這是通過作為實際業務邏輯「瘦」包裝的 Action類完成的。

  • Model(模型)的狀態
    模型表示應用程序的狀態。業務對象更新應用程序的狀態。ActionForm bean在會話級或請求級表示模型的狀態,而不是在持久級。JSP 文件使用 JSP標記讀取來自 ActionForm bean 的信息。

  • View(視圖)
    視圖就是一個 JSP文件。其中沒有流程邏輯,沒有業務邏輯,也沒有模型信息 --只有標記。標記是使 Struts 有別於其他框架(如Velocity)的因素之一。

詳細分析 Struts
圖 6 顯示的是 org.apache.struts.action 包的一個最簡UML 圖。圖 6 顯示了 ActionServlet(Controller)、ActionForm (Form State) 和Action (Model Wrapper) 之間的最簡關係。


圖 6. Command (ActionServlet)與 Model (Action & ActionForm) 之間的關係的 UML圖
ActionServlet 與 Action 和 ActionForm 的關係

ActionServlet 類
您還記得函數映射的日子嗎?在那時,您會將某些輸入事件映射到一個函數指針上。如果您對此比較熟悉,您會將配置信息放入一個文件,並在運行時加載這個文件。函數指針數組曾經是用C 語言進行結構化編程的很好方法。

現在好多了,我們有了 Java 技術、XML、J2EE,等等。Struts的控制器是將事件(事件通常是 HTTP post)映射到類的一個servlet。正如您所料 --控制器使用配置文件以使您不必對這些值進行硬編碼。時代變了,但方法依舊。

ActionServlet 是該 MVC 實現的 Command部分,它是這一框架的核心。ActionServlet (Command)創建並使用 Action、ActionForm 和ActionForward。如前所述,struts-config.xml文件配置該 Command。在創建 Web 項目時,您將擴展 Action和 ActionForm 來解決特定的問題。文件struts-config.xml 指示 ActionServlet如何使用這些擴展的類。這種方法有幾個優點:

  • 應用程序的整個邏輯流程都存儲在一個分層的文本文件中。這使得人們更容易查看和理解它,尤其是對於大型應用程序而言。

  • 網頁設計人員不必費力地閱讀 Java代碼來理解應用程序的流程。

  • Java 開發人員也不必在更改流程以後重新編譯代碼。

可以通過擴展 ActionServlet 來添加 Command功能。

ActionForm 類


ActionForm 維護 Web應用程序的會話狀態。ActionForm是一個抽象類,必須為每個輸入表單模型創建該類的子類。當我說 輸入表單模型 時,是指ActionForm 表示的是由 HTML表單設置或更新的一般意義上的數據。例如,您可能有一個由 HTML表單設置的 UserActionForm。Struts框架將執行以下操作:

  • 檢查 UserActionForm是否存在;如果不存在,它將創建該類的一個實例。

  • Struts 將使用 HttpServletRequest 中相應的域設置UserActionForm 的狀態。沒有太多討厭的request.getParameter() 調用。例如,Struts框架將從請求流中提取 fname,並調用UserActionForm.setFname()。

  • Struts 框架在將 UserActionForm 傳遞給業務包裝UserAction 之前將更新它的狀態。

  • 在將它傳遞給 Action 類之前,Struts 還會對UserActionForm 調用 validation()方法進行表單狀態驗證。 註: 這並不總是明智之舉。別的網頁或業務可能使用UserActionForm,在這些地方,驗證可能有所不同。在UserAction 類中進行狀態驗證可能更好。

  • 可在會話級維護 UserActionForm。

註:

  • struts-config.xml 文件控制 HTML 表單請求與ActionForm 之間的映射關係。
  • 可將多個請求映射到 UserActionForm。
  • UserActionForm可跨多頁進行映射,以執行諸如嚮導之類的操作。

Action 類
Action 類是業務邏輯的一個包裝。Action類的用途是將 HttpServletRequest 轉換為業務邏輯。要使用Action,請創建它的子類並覆蓋 process()方法。


ActionServlet (Command) 使用 perform()方法將參數化的類傳遞給 ActionForm。仍然沒有太多討厭的request.getParameter()調用。當事件進展到這一步時,輸入表單數據(或 HTML表單數據)已被從請求流中提取出來並轉移到 ActionForm類中。

註:擴展 Action 類時請注意簡潔。Action類應該控制應用程序的流程,而不應該控制應用程序的邏輯。通過將業務邏輯放在單獨的包或EJB 中,我們就可以提供更大的靈活性和可重用性。

考慮 Action 類的另一種方式是 Adapter設計模式。Action的用途是「將類的接口轉換為客戶機所需的另一個接口。Adapter使類能夠協同工作,如果沒有Adapter,則這些類會因為不兼容的接口而無法協同工作。」(摘自 Gof所著的 Design Patterns - Elements of Reusable OOSoftware )。本例中的客戶機是ActionServlet,它對我們的具體業務類接口一無所知。因此,Struts提供了它能夠理解的一個業務接口,即 Action。通過擴展Action,我們使得我們的業務接口與 Struts業務接口保持兼容。(一個有趣的發現是, Action是類而不是接口)。Action開始為一個接口,後來卻變成了一個類。真是金無足赤。)

Error 類
UML 圖(圖 6)還包括 ActionError 和ActionErrors。ActionError封裝了單個錯誤消息。ActionErrors 是ActionError 類的容器,View可以使用標記訪問這些類。ActionError 是 Struts保持錯誤列表的方式。

圖 7. Command (ActionServlet)與 Model (Action) 之間的關係的 UML 圖

ActionServlet 與 Action 的關係

ActionMapping 類
輸入事件通常是在 HTTP 請求表單中發生的,servlet 容器將 HTTP請求轉換為HttpServletRequest。控制器查看輸入事件並將請求分派給某個Action 類。struts-config.xml 確定Controller 調用哪個 Action類。struts-config.xml 配置信息被轉換為一組ActionMapping,而後者又被放入ActionMappings 容器中。(您可能尚未注意到這一點,以 s 結尾的類就是容器)

ActionMapping 包含有關特定事件如何映射到特定Action 的信息。ActionServlet (Command)通過 perform() 方法將 ActionMapping傳遞給 Action 類。這樣就使 Action可訪問用於控制流程的信息。

ActionMappings
ActionMappings 是 ActionMapping對象的一個集合。

再訪郵件列表樣例
下面我們看一下 Struts 是如何解決困擾 join.jsp的這些問題的。改寫後的方案由兩個項目組成。第一個項目包含應用程序的邏輯部分,這個應用程序是獨立於Web 應用程序的。這個獨立層可能是用 EJB技術實現的公共服務層。為了便於說明,我使用 Ant構建進程創建了一個稱為 business的包。有幾個原因促使我們使用獨立的業務層:

  • 劃分責任
    單獨的包使管理人員能夠在開發小組內委派責任。這也有助於提高開發人員的責任心。

  • 通用件

    我們設想開發人員將這個包看作一個商業軟件。將它放在另外的包中使它更像通用件。這個包可能是通用件,也可能是由組織內部的另一個小組開發的。

  • 避免不必要的構建和單元測試。
    分開的構建進程有助於避免不必要的構建和單元測試。

  • 使用接口開發
    在進行開發和避免不必要的耦合時,它有助於從接口的觀點來思考問題。這是極重要的一個方面。當開發您自己的業務包時,這些業務類不應該關心到底是Web應用程序執行調用,還是獨立應用程序執行調用。因此,應該避免在業務邏輯層使用對servlet API 或 Struts API 調用的任何引用。

  • 穩定性
    並不是每個組織都每天、每週甚至每月進行檢修。因此,在進行開發時,穩定的接口點是重要的。不能因為業務包處於變遷階段就認為Web 項目也應該處於變遷階段。

業務構建註釋
我用 Ant 構建項目,並用 JUnit 運行單元測試。business.zip包含構建業務項目所需的一切,當然 Ant 和 JUnit除外。這個包腳本將構建類,運行單元測試,創建 Java 文檔和 jar文件,最後將所有這些內容壓縮到一個 zip 文件中發送給客戶。只要對build.xml作一些修改,您就可以將它部署到其他平台上。Business.jar位於 Web 的下載部分,因此,您並非必須下載並構建這個業務包。

Web 項目
第二個項目是用 Struts 開發的一個 Web 應用程序。您將需要一個符合 JSP1.1 和 Servlet 2.2 規範的容器。最快的入門方法是下載並安裝 Tomcat3.2(請參閱 參考資源 )。直到有 Struts 的1.0 發行版之前,我建議您從 Jakarta 項目獲得最新的版本(請參閱 參考資源
)。這對我來說是個大問題,我不能確保我的Web 項目樣例能與您下載的 Struts 一起工作。Struts仍在不斷變化,所以我不得不經常更新我的項目。在本項目中,我使用的是jakarta-struts-20010105.zip。圖 8 顯示了此 Web項目的結構。如果您已安裝了 Ant,則運行這個版本將創建一個稱為joinStruts.war 的 war文件,您隨時可以部署這個文件。

圖 8. Web項目的結構
Web 項目的結構

清單 4 顯示了轉換後的 JSP 文件,稱為joinMVC.jsp。這個文件從最初的 50 行變為 19行,並且現在不含任何 Java代碼。從網頁設計人員的角度來看,這是個巨大的改進。

清單 4. joinMVC.jsp -- 再訪簡單的 JSP

            <%@ page language="java" %>
              <%@ taglib uri="/WEB-INF/struts.tld" prefix="struts" %>
                <%@ taglib uri="/WEB-INF/struts-form.tld" prefix="form" %>
                  <html>
                    <head>
                      <title>
                        <struts:message key="join.title"/>
                      </title>
                    </head>
                    <body bgcolor="white">
                      <form:errors/>
                      <h3>Enter your email to join the group</h3>
                      <form:form action="join.do" focus="email" >
                        <form:text property="email" size="30" maxlength="30"/>
                        <form:submit property="submit" value="Submit"/>
                        </form:form>
                    </body>
                  </html> 

網頁的變化
下面是使用 Struts 標記庫之後所發生變化的列表:

  • Import
                        <%@ taglib uri="/WEB-INF/struts.tld" prefix="struts" %>
                          
    用於 Java 代碼的<%@page import? 已被替換為用於Struts 標記庫的 <%@ taglib uri?。

  • 文本
                            <struts:message key="join.title"/>
                            
    資源屬性文件包含 join.title的文本。在本例中,ApplicationResources屬性文件包含這個名值對。這使字符串更易於查看和國際化。

  • 錯誤
                            <form:errors/>
                            
    ActionServlet 或 ActionForm構建要顯示的錯誤消息。這些錯誤消息也可以包含在屬性文件中。ApplicationResources也提供了一種格式化錯誤的方法,即設置 error.header 和error.footer。

  • HTML 表單
                            <form:form action="join.do" focus="email" >
                              
    • JSP<form> 標記和屬性替代了 HTML<form> 標記和屬性。<formaction="join.jsp" name="join"> 已更改為<form:form action="join.do" focus="email"> 。
    • HTML 標記已替換為<form:text/> 。
    • HTML 標記已替換為<form:submit/> 。

模型 -- 會話狀態
JoinForm 擴展了 ActionForm并包含表單數據。本例中的表單數據只有電子郵件地址。我已為電子郵件地址添加了一個寫方法和讀方法,以供框架訪問。為了便於說明,我重寫了validate() 方法,並使用了 Struts 的跟蹤功能。Struts將創建 JoinForm 並設置狀態信息。

模型 -- 業務邏輯
如前所述,Action是控制器和實際業務對象之間的接口。JoinAction 包裝了對business.jar 的調用,這些調用最初在join.jsp 文件中。JoinAction 的perform() 方法在清單 5 中列表。

清單 5. - JoinAction.perform()

public ActionForward perform(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) 
throws IOException, ServletException { // 抽取我們將會用到的屬性和參數 
JoinForm joinForm = (JoinForm) form; 
String email = joinForm.getEmail(); 
ActionErrors errors = new ActionErrors(); // 存儲輸入.... 
try { business.db.MailingList.AddEmail(email); } 
catch (Exception e) { // 記錄日誌,打印棧 // 將錯誤回顯給用戶 
errors.add("email",new ActionError("error.mailing.db.add")); } 
// 如需任何消息,請將指定的錯誤消息鍵保存到 //HTTP 請求中,以供<struts:errors> 標記使用。 
if (!errors.empty()) { saveErrors(request, errors); // 返回到初始表單 
return (new ActionForward(mapping.getInput())); } // 將控制權轉交給 Action.xml 中指定的 'success' URI 
return (mapping.findForward("success")); } 

註:perform() 返回一個稱為ActionForward的類,該類通知控制器下一步該執行什麼操作。在本例中,我使用從控制器傳入的映射來決定下一步的操作。

控製器
我已修改了 JSP文件,並創建了兩個新類:一個類用來包含表單數據,一個類用來調用業務包。最後,我通過修改配置文件struts-config.xml 將它們整合起來。清單 6顯示了我添加的 action 元素,這個元素用來控制joinMVC.jsp 的流程。

清單 6. Action 配置


                    <actionpath="/join"  name="joinForm"  type="web.mailinglist.JoinAction" 
                    scope="request" input="/joinMVC.jsp"  validate="true">
                    <forwardname="success"path="/welcome.html"/>
                    </action> 

action 元素描述了從請求路徑到相應的 Action類的映射,應該用這些類來處理來自這個路徑的請求。每個請求類型都應該有相應的action 元素,用來描述如何處理該請求。對於 join 請求:

  1. joinForm 用來容納表單數據。
  2. 因為 validate 被標記為 true,所以 joinForm將試圖進行自我驗證。
  3. web.mailinglist.JoinAction是用來處理對這個映射的請求的 action 類。
  4. 如果一切順利,該請求將轉到 welcome.jsp。
  5. 如果出現業務邏輯故障,流程將返回到joinMVC.jsp,這是最初發出請求的網頁。為什麼會這樣呢?在清單6 的 action 元素中,有一個稱為 input 的屬性,其值為"/joinMVC.jsp"。在我的JoinAction.perform()(如清單 5所示)中,如果業務邏輯失敗,perform() 就返回一個ActionForward,並以 mapping.getInput()作為參數。本例中的 getInput() 是"/joinMVC.jsp"。如果業務邏輯失敗,它將返回到joinMVC.jsp,這是最初發出請求的網頁。

使用 Struts 前後的比較
正如我們在圖 9 中所看到的那樣,複雜性和層都有顯著增加。不再存在從JSP 文件到 Service 層的直接調用。

圖 9. 使用 Struts前後的比較
使用 Struts 前後的比較

Struts 的優點

  • JSP 標記機制的使用

    標記特性從 JSP 文件獲得可重用代碼和抽象 Java代碼。這個特性能很好地集成到基於 JSP的開發工具中,這些工具允許用標記編寫代碼。

  • 標記庫
    為什麼要另發明一種輪子,或標記庫呢?如果您在庫中找不到您所要的標記,那就自己定義吧。此外,如果您正在學習JSP 標記技術,則 Struts 為您提供了一個起點。

  • 開放源碼
    您可以獲得開放源碼的全部優點,比如可以查看代碼並讓使用庫的每個人檢查代碼。許多人都可以進行很好的代碼檢查。

  • MVC 實現樣例
    如果您希望創建您自己的 MVC 實現,則 Struts 可增加您的見識。

  • 管理問題空間
    分治是解決問題並使問題可管理的極好方法。當然,這是一把雙刃劍。問題越來越複雜,並且需要越來越多的管理。

Struts 的缺點

  • 仍處於發展初期
    Struts 開發仍處於初級階段。他們正在向著發行版本 1.0而努力,但與任何 1.0 版本一樣,它不可能盡善盡美。

  • 仍在變化中
    這個框架仍在快速變化。Struts 1.0 與 Struts 0.5相比變化極大。為了避免使用不贊成使用的方法,您可能隔一天就需要下載最新的Struts。在過去的 6 個月中,我目睹 Struts 庫從 90K 增大到 270K以上。由於 Struts中的變化,我不得不數次修改我的示例,但我不保證我的示例能與您下載的Struts 協同工作。

  • 正確的抽象級別

    Struts是否提供了正確的抽象級別?對於網頁設計人員而言,什麼是正確的抽象級別呢?這是一個用$64K的文字才能解釋清楚的問題。在開發網頁的過程中,我們是否應該讓網頁設計人員訪問Java 代碼?某些框架(如 Velocity)說不應該,但它提供了另一種 Web開發語言讓我們學習。在 UI 開發中限制訪問 Java有一定的合理性。最重要的是,如果讓網頁設計人員使用一點Java,他將使用大量的 Java。在 Microsoft ASP的開發中,我總是看到這樣的情況。在 ASP 開發中,您應該創建 COM對象,然後編寫少量的 ASP 腳本將這些 COM 對象聯繫起來。但是,ASP開發人員會瘋狂地使用 ASP 腳本。我會聽到這樣的話,「既然我可以用VBScript 直接編寫 COM 對象,為什麼還要等 COM開發人員來創建它呢?」通過使用標記庫,Struts 有助於限制 JSP文件中所需的 Java 代碼的數量。Logic Tag就是這樣的一種庫,它對有條件地生成輸出進行管理,但這並不能阻止 UI開發人員對 Java代碼的狂熱。無論您決定使用哪種類型的框架,您都應該了解您要在其中部署和維護該框架的環境。當然,這項任務真是說起來容易做起來難。

  • 有限的適用範圍
    Struts 是一種基於 Web 的 MVC 解決方案,所以必須用 HTML、JSP 文件和servlet 來實現它。

  • J2EE 應用程序支持
    Struts 需要支持 JSP 1.1 和 Servlet 2.2 規範的 servlet容器。僅憑這一點遠不能解決您的全部安裝問題,除非使用 Tomcat3.2。我用 Netscape iPlanet 6.0安裝這個庫時遇到一大堆問題,按理說它是第一種符合 J2EE的應用程序服務器。我建議您在遇到問題時訪問 Struts用戶郵件列表的歸檔資料(請參閱 參考資源 )。

  • 複雜性
    在將問題分為幾個部分的同時也引入了複雜性。毫無疑問,要理解 Struts必須接受一定的培訓。隨著變化的不斷加入,這有時會令人很沮喪。歡迎訪問本網站。

  • 在何處...

    我還能指出其他問題,例如,控制器的客戶端驗證、可適用工作流程和動態策略模式在什麼地方?但是,目前這太容易成為吹毛求疵的問題,有些問題是無關緊要的,或者說應該對1.0 發行版提這些問題。隨著 Struts 小組的不斷努力,到您閱讀本文時Struts 說不定已經有了這些功能,或者它很快就會具有這些功能。

Struts 的前景
在這個軟件開發的新時代,一切都變得很快。在不到 5年的時間內,我已經目睹了從 cgi/perl 到 ISAPI/NSAPI、再到使用 VB 的ASP、一直到現在的 Java 和 J2EE 的變遷。Sun 正在盡力將新的變化反映到JSP/servlet 體系結構中,正如他們對 Java 語言和 API所作的更改一樣。您可以從 Sun 的網站獲得新的 JSP 1.2 和 Servlet 2.3規範的草案。此外,一個標準 JSP標記庫即將出現;有關這些規範和標記庫的鏈接,請參閱 參考資源

最後的註釋
Struts 使用標記和 MVC解決了某些重大問題。這個方法有助於提高代碼的可重用性和靈活性。通過將問題劃分為更小的組件,當技術空間或問題空間中出現變化時,您就有更多的機會重用代碼。此外,Struts使網頁設計人員和 Java開發人員能將精力集中於自己最擅長的方面。但是,在強健性增強的同時,也意味著複雜性的增加。Struts比簡單的單個 JSP 網頁要複雜得多,但對於更大的系統而言,Struts實際上有助於管理複雜性。另外,我並不想編寫自己的 MVC實現,而只想了解一個這樣的實現。不管您是否會使用 Struts,回顧這個Struts 框架(對不起,應該是庫)都會使您對 JSP 文件和 servlet的特性、以及如何將它們組合起來用於您的下一個 Web項目有更好的了解。正像翼間支柱是機翼結構中不可缺少的一部分一樣,Strut也可能成為您下一個 Web 項目的不可缺少的一部分。


參考資源

作者簡介
Malcolm G. Davis住在阿拉巴馬州伯明翰市,他在自己的諮詢公司當總裁。他自稱是一名 Java傳道者。他在宣傳 Java的優點的閒暇之餘,他會去長跑,或者與自己的孩子一起玩。可以通過 malcolm@nuearth.com 與Malcolm 聯繫。

冷日
(冷日)
Webmaster
  • 註冊日: 2008/2/19
  • 來自:
  • 發表數: 15771
[分享]為 Web 服務構建 Struts 應用程序
在 Web 服務領域引入 MVC 模式的功能
當您要將企業應用程序轉變成與 Web服務一起使用時,最簡單的途徑就是將單個操作與單個的企業服務結合在一塊。但這並不是最好的方法。在本文中,Jerome Josephraj將向您展示如何構架基於分層的、正確的模型-視圖-控制器(MVC)設計模式的Web 服務應用程序。為此,他修改了Struts(一種流行的開放源碼MVC 框架),以使其適用於 Web 服務領域。通過研究此處所概述的樣本應用程序,您將知道如何才能將Struts 與 Web 服務聯合起來使用。

不斷發展的 Java 編程語言和 Sun 公司的 J2EE規範使得遵守各類準則的軟件開發者們能夠創建出分布式計算應用程序,這些應用程序在以前只能通過相關專門工具才可實現。這樣,當某些開發團體要選擇在 Java平台中實現新系統時,其他團體就會通過另外的技術來創建、提高並維護應用程序,然後將它們集成到已有的各類分布式應用程序中去。這種情形就引起了互操作性的競爭。新應用程序與舊應用程序如何交互呢?答案就是:Web服務。Web 服務是程序設計新的聖杯。它們能夠共享並協調分散的各類計算資源。

在本文中,您將了解實現此目的的一種方法。您將看到如何在與 Web服務相結合的開放源碼 Struts框加的基礎上來構建應用程序。在開始之前您必須了解一些 J2EE 與 Web 服務的相關知識。這裡,我將簡要地介紹下 Struts架構與模型-視圖-控制器(MVC),如果您以前沒碰到過,那麼您應該在後面先看下 參考資料部分來獲取更多信息。

MVC 模式:分離開發角色

MVC設計模式很清楚的劃定了程序員與設計者的角色界限。換句話說,從商業邏輯上拆解了數據。這種模式是讓設計者集中於設計應用程序的顯示部分,而開發者則集中於開發驅動應用程序功能所需的組件。

MVC 模式有好幾種變異,不過它們都是基於相同的基礎結構:應用程序的數據模型 (Model),顯示層代碼(View),以及程序控制邏輯( Controller)是存在其中的獨立但能相互間通信的組件。模型組件描述並處理應用程序數據。視圖指的是用戶界面;它反映的是模型數據並把它遞交給用戶。控制器是將視圖上的行為(例如,按下Submit 按鈕)映射到模型上的操作(例如,檢索用戶詳細信息)。模型更新後,視圖也被更新,用戶就能夠完成更多行為。MVC模式使代碼易懂而且使代碼更容易重用;另外,在很多工程中視圖經常要被更新,MVC 模式將模型和控制器與這些所做的更改獨立開來。

圖 1 MVC 模式概要

圖 1. MVC 設計模式
MVC 設計模式

請參閱下面的參考資料,以獲取更多關於模型-視圖-控制器的更多信息。

Struts:基於 MVC 的堅固框架

Struts是 MVC 模式基礎上構建 Web 應用程序的一種開放源碼框加。Struts 鼓勵在 MVC 模式上構建應用程序而且提供大多數 Web 應用程序所共有的服務。

在 Struts應用程序中,您可以構建模型層,這樣業務邏輯與數據檢索邏輯重用就很容易了。這層負責運行應用程序的業務邏輯,獲取相關數據(例如,運行SQL 命令或者讀取平面文件)。

Struts 鼓勵在模型-視圖-控制器設計範例基礎上構建應用程序。Struts 提供自己的控制器組件(ActionController類)並與其他技術相結合來提供模型與視圖。對於模型(Model 類), Struts 能與任何標準的數據訪問技術相結合,包括 EJB、JDBC以及 Object-Relational Bridge。對於視圖(ActionForm類),Struts 在 JSP 環境以及其他基於表示邏輯的系統中運行的很好。圖 2 闡明了基於Struts 應用程序的邏輯流程。

圖 2. Struts 應用程序的邏輯流
Struts 應用程序的邏輯流程

簡單粗糙的 Web 服務體系結構

構建Web 服務最簡單的途徑就是將單個操作與單個企業服務結合起來,如圖 3所示。在這種設計方法中,實現業務邏輯的服務與實現數據檢索的服務是混合在一起的。

圖 3. 簡單粗糙的 Web 服務體系結構
簡單粗糙的 Web 服務體系結構

這樣的一種 Web服務可以很容易從已有的業務組件中開發出來。然而,它有很多的缺點:對用戶沒有提供統一認證,提供者與訂閱者耦合不緊,業務邏輯沒有重用。簡而言之,對於一個連貫的解決方案來說這並不是一個非常好的體系結構。

在 MVC 模式基礎上來實現 Web 服務解決方案會更好點。在後續部分,您將看到如何使用 Struts 來做到這一點。我將通過 WSManager層來詳述現有的 Struts 架構,這一層展示了採用 Web 服務的模型服務方法。

採用 Struts 應用程序的 Web 服務

在以後的開發工作中您可以擴大構建得比較好的 Struts 應用程序來支持 Web服務。前面講過,Struts架構清晰地區分開了視圖、控制器以及模型。模型包含所有必須的業務邏輯,從存儲數據倉庫中檢索數據。您可以構建一個簡單的 Web 服務層——稱為 WSManager層——這樣的模型可以提供 Web 服務也可以訂購 Web 服務。使用這種體系結構的應用程序將基於組件開發的最佳點與萬維網結合在了一起,如圖 4所示。

圖 4. 採用 Web 服務的 Struts 應用程序
採用 Web 服務的 Struts 應用程序

下面的部分要討論的是這種體系結構中不同組件的詳細信息,特別要密切注意 WSManager層,因為它是此體系結構中真正新加的部分。

Struts 控製器
MVC體系結構的控制器部分主要集中於接收客戶端的請求(一般為運行 Web瀏覽器的用戶),決定執行哪一種業務邏輯功能來響應請求,然後負責生成下一個用戶界面連接到合適的視圖組件上去。在 Struts中,控制器的主要組件就是 ActionServlet 類的一個小服務程序。

ActionServlet 負責通過 XML 文檔將 URI 請求映射到特定的行為。這個文檔包含了URI 請求列表而且還告知 ActionServlet 它應該如何分配每個請求。這種方法有幾個好處:

  • 應用程序的整個邏輯流程在分級文本文件中。
  • 這種格式的列表更容易查看與理解,尤其是對於一個很大的應用程序而言。

ActionServlet 決定了應用程序的流程。許多 Action類都繼承了 ActionServlet 。每個 Action類:

  • 都映射到各自獨立的處理過程
  • 通過 Struts 的 ActionController與 Struts JSPs 相結合
  • 作為繼承 Struts中 Action 類的一個 Java 類來實現。

Struts中 Action 類調用 WSManager 類中的相關方法來使用 Web服務。 WSManager 獲取所要求的響應——或者如果有一個被解除了就會出現異常——將它回傳給 Struts控制器。

WSManager
WSManager接收 JAX-RPC 端點的請求。將 WSManager 類中的方法調用映射到新來的客戶端請求。這些新來的客戶端請求是SOAP 信息的格式。WSManager必須實現安全認證,轉變參數,在指定請求到模型服務之前,要對這些請求進行參數預處理。請求所包含的參數形式有 Java對象,Java原始參數,XML 文檔,或者甚至是 SOAP 文檔分片(例如,SOAP Element對象)。這些類型必須要轉化成內部所支持的 schema(例如,預定義的 Java 數據訪問對象)。

雖然 WSManager 可以很直接地處理與 Java 對象結合的參數,但它還需要採取一些附加步驟來處理 XML文檔。建議以下步驟:

  1. WSManager 類應該能夠通過XML Schema來驗證XML文檔的有效性。
  2. WSManager 類接著要把 XML 文檔轉化成內部支持的 schema。
  3. 最後 WSManager 應該分解文檔並且儘可能地將它映射到域對象中去。

WSManager 實現以下任務是非常重要的:

  • 身份驗證和授權使用
  • 錯誤處理
  • 緩存。

WSManager 也可以生成響應;這個過程由方法調用返回值的簡單構造所組成。WSManager 中保留這樣的功能,您可以通過緩存數據從而避免重複訪問模型服務層。您還可以集中管理響應聚集以及XML 文檔轉換,如果您要返回給調度者的文檔必須遵守與內部 schema 不同的 schema 時,這一點就顯得尤其重要。

WSManager處理所有SOAP 請求並把它們委派到模型層所暴露的業務邏輯。如果模型服務是作為一個 EJB 層來實現,那您一樣可以在 EJB 層中通過 SessionFac,ade 設計模式來實現。如果您採用這種模式執行 WSManager ,您將獲得很多好處,因為 WSManager會:

  • 作為初始聯繫點來管理請求與服務
  • 調用安全服務,包括身份驗證與授權使用,從而避免任何重複的層訪問。
  • 委派業務處理(採用由 Struts 應用程序所使用的模型服務)
  • 在 WSManager層緩存數據避免任何不必要的數據庫訪問。

發布者:展示 Web 服務
在 WSManager類中實現的每一個公共方法都將作為一種 Web 服務發布出來。換句話說,您要為這些類發布一種Web 服務描述 。Web 服務描述是由服務的 Web 服務描述語言(WebServices Description Language,WSDL)描述與由它所引用的任何 XML schema所組成。(WSDL 是描述服務的標準語言。)

您可以在公共註冊中心或在企業內的公司註冊表上發布 Web 服務描述。同樣,你也可以發布由WEB 服務定義的XMLSchema到同一個公共的或者公司專有的(UDDI)註冊中心上。 Java Web 服務客戶端採用 JAXR應用編程接口來查詢公司或公共註冊表上的服務描述。

如果您的客戶都是專有的合作夥伴,那您就不需要使用註冊表了。不過,您可以在您的應用程序的 Web 層或者在具備適當安全保護的熟知位置來發布您的Web 服務描述( WSDL 和 XML schemas)。例如,假想有一個轉售者的客戶應用程序,他與某個特殊廠商有協定。客戶應用程序已經在廠商開發時間中靜態地與 Web 服務結合在一起了。只有授權的團體才可以查詢 XML schema 或者從 Web 層檢索服務描述來生成客戶端代碼。您應該在 WSManager層中執行有效客戶的身份驗證和授權使用。

訂購者:使用 Web 服務
應用程序可以利用在公共註冊表或者企業內部中已存的 Web 服務。 WSManager有解析必要的 WSDL文件的方法並且調用相關的操作返回一個值。Struts 控制器在 WSManager 類中調用相關方法來使用特定的Web 服務。數據作為預定義的數據訪問對像在 WSManager 和 Struts 控制器之間來回傳送。訪問 Web服務時所發生的所有異常都將在 WSManager中列舉出來並且傳回到 Struts ActionController.

服務請求者要通過使用服務代理來搜索 Web 服務;如果找到它所想要使用的 Web 服務,為使用這個服務它將與服務提供者建立一個合約,然後才能調用服務中的業務。

WSManager使用 WSDL文檔、服務器名字、端口名、操作名以及包括 Java原始類型、Java 數組、Java 對象 或者與 XML 文檔等所有必需的請求參數一起來訂購一個 Web 服務。

如果是在 UDDI 註冊表中發布目標 Web 服務,那麼所有基於 Struts 的應用程序都能使用像 XMethods(請參閱參考資料)這樣的代理服務來訂購它。在執行完所請求的操作後,供應者 Web 服務返回所期望的值。 WSManager可以更改返回值使得它與應用程序所期待的 schema 相匹配;它也可以在應用程序要求基礎上修改結果。從 WSManager收到結果以後,Struts ActionController能夠處理結果並傳送給相關視圖,或者還能調用相關的模型服務來完成深層處理。

錯誤處理
所有的錯誤都是在 WSManager 層中處理的;這就消除了不必要的服務器開銷。如果模型服務是當作 EJB層來實現的,那麼其性能會得到非常顯著地增強。

在充當供應者角色時, WSManager拋出所有如 SOAPFaultException這樣的異常。它還可以檢查新來的請求並且拋出所有缺少強制字段的異常。您可以創建一個類來跟蹤並在數據倉庫中記錄這些錯誤,以後可以做作參考。

在充當訂購者角色時, WSManager捕捉到由服務供應者所拋出的所有 SOAP 異常並且將它們更改為 WSManager所要求的格式。您可以記錄下所拋出的錯誤作為以後參考之用。如果需要,那麼在需要時候也能夠校驗出響應值並且作為異常拋出。您可以創建一個類來記錄這些異常,作為以後參考之用。 WSManager能夠驗證響應值而且可以作為一個異常將它拋出。

審核
在充當供應者角色時, WSManager可以記錄下詳細信息,將來可以做為審核。使用這些信息有很多用途,像:

  • 在所接收到的大量採樣點基礎上登記客戶端。
  • 收集數據用作市場目的。
  • 決定應用程序是否需要更新。
  • 鑑別並捕捉非法用戶。

緩存
Web 服務的客戶往往要比一般的客戶端-服務器體系結構中的客戶要多些;因此在 Web服務體系結構中,客戶端就要做更多的工作,比如緩存。Web服務正確使用數據緩存就可以實現其最大的性能。當服務的請求信息主要是只讀的時候或者當那些信息按照比所要求的速率變化得還要慢時,您就應該考慮要在 Web服務中使用緩存了。

身份驗證與授權使用
您可以在 WSManager層中執行所有訂購者的身份驗證。所有想使用 Web服務的客戶都要經過這樣的身份驗證邏輯。您可以使用基本用戶身份驗證或者數字證書來實現此目的。

Struts 視圖
您要通過使用JSP 技術來構建基於 Struts 應用程序的視圖部分。JSP 頁包含有靜態 HTML 加上動態內容,這些內容是基於對特別行為標籤說明(在頁面要求時)的。JSP 環境包括一套標準的行為標籤。另外,在自定義標籤庫中組織了一套標準的工具,開發者們可以使用這些工具來定義他們自己的標籤。

Struts框架包含有擴展的自定義標籤庫,這個庫能幫助用戶界面國際化更為全面並能非常適度地與 ActionForm組件相互作用。視圖層比較單薄,它不提供業務邏輯。Struts 視圖是通過 ActionForm 與 Struts控制器相互作用的。

Struts ActionForm
ActionForms只是一些 Java 類而已,它繼承了 Struts 所提供的 ActionForm 類,這些類中包含有 accessor和 mutator 方法。 JSP 頁或者 Action 類都會調用這些方法來檢索或者從數據庫刷新數據。

模型服務
模型服務是作為一組Java 類來執行的。每個模型服務組件都會提供一套服務,而這些組件結合起來同樣也提供一套普通服務。 ActionController與 WSManager類將數據當作預定義的數據訪問對象來回傳送。在處理過程中,ActionController或者 WSManager可以調用相關模型服務組件中所要求的方法。這些組件將所要求的數據以數據訪問對象的形式傳給模型服務,模型服務執行一切必須的商業邏輯處理然後從存儲數據倉庫中取出所需要的數據。模型服務組件聚集相關的預定義數據訪問對象,然後將它傳回給 ActionServlet或者 WSManager類。所有的錯誤或者確認信息都會通知給 ActionServlet或 WSManager層。

對於您的應用程序,您應該遵守以下的設計規則:

  • 模型服務不能含有任何與視圖相關的代碼(例如,會話處理)。
  • 所有的事務僅僅只能在 Action 或 WSManager層中來提交。
  • 模型服務只能被同一模型服務組件中其他的模型服務或者高層 Action 類所調用。

數據存儲層
數據存儲層由所有的存儲數據倉庫組成。例如,它可能包含有關係數據庫、平面文件或者甚至有 XML 文檔。

示例

我已經附上了實現此處所討論的體系結構的一個簡單示例。所有的示例代碼都包含在了 zip文件中(請參閱下面的參考資料)。這個例子舉例說明了一個簡單的新聞Portal。新聞內容是從數據源(這裡稱為 DataSource)傳送到 JSP 頁,同時信息內容也要作為一個 Web服務發布出來。這個 Portal 也可以從 StockQuote Web服務中檢索最新的股票報價並顯示在同一個 JSP 頁中。

這個 zip 文件包含了所有的源代碼和 DataSource 的 SQL 腳本。它還包含有 Struts Action和 ActionForm類、模型服務類以及用來為發布者與訂購者顯示結果及其源代碼的 JSP 頁。它同時還包含有WSDL 文件以及客戶端源代碼來訪問已發布的 NewsContent Web 服務。

這裡我不再闡述示例代碼的詳細信息了;您想要發現它的複雜性,最好的途徑就是自己去試驗。開始時,我會向您說明應用程序如何將信息內容(存儲在 DataSource中)和股票信息(從 Web 服務處獲得)遞交給 JSP 頁的。

JSP 頁被加載時,它就調用 Action 類,這個類在 MVC 設計模式中充當控制器角色。 Action類通過傳遞預定義的數據訪問對象來調用模型服務類中的 getNews() 方法。(清單 1中所示的一小段 Action 類代碼說明了如何從 Action 類中調用模型服務。)

getNews()方法實現了所有必需的業務邏輯與數據檢索邏輯。一旦從數據源取出了相關數據,模型服務就可以聚集數據訪問對象並將它發送回給 Action類。

清單 1. Action 類的部分代碼
/*
 * Create a pre-defined Data Access Object
 */
newsSearchResultVOB = new NewsSearchResult();
/*
 * Call Business layer NewsMs's searchNews method passing the NewsCOD
 * and get back a Collection of News
 */
newsList = (Vector) newsMs.getNews(newsSearchResultVOB);
/*
 * Put the NewsList in the request.
 *
 */
request.setAttribute(SystemConstants.NEWS_LIST, newsList);
//Set the newslist in the form
newsFrm.setNewsDetails(newsList);
// Set the NewsForm in the request
request.setAttribute(mapping.getAttribute(), newsFrm);
//return actionForward;
return mapping.findForward("success");

從模型服務處接收到數據訪問對象後, Action類通過傳遞轉給模型服務的同一個數據訪問對象來調用 WSManager類中的 getStocks() 方法。 WSManager類中的 getStocks()方法執行 JAX-RPC 來訂購股票 Web 服務。

從股票 Web 服務獲得的結果要轉變成本地的 schema 並且聚集回到預定義數據訪問對象中去。這個對象從它被調用的地方傳回到 action類。 清單 2 闡明了如何使用 WSManager中的 JAX-RPC來訂購 Web 服務,以及如何將股票 Web 服務的結果轉變成本地 schema 並聚集回到數據訪問對象中去。

清單 2. 訂購 Web 服務及傳送響應

public Vector getStocks() throws Exception {
//Method level variables
Vector stockValues = new Vector();
try {
// create service factory
javax.xml.rpc.ServiceFactory factory =
javax.xml.rpc.ServiceFactory.newInstance();
// define targetNameSpace
String targetNamespace = "http://www.themindelectric.com/"
+ "wsdl/net.xmethods.services.stockquote.StockQuote/";
 // define qname
QName serviceName =
new QName(targetNamespace,
"net.xmethods.services.stockquote.StockQuoteService");
 // define portname
QName portName =
new QName(targetNamespace,
"net.xmethods.services.stockquote.StockQuotePort");
 // define operation name
QName operationName = new QName("urn:xmethods-delayed-quotes",
"getQuote");
//Specify wsdl location
java.net.URL wsdlLocation =
new java.net.URL("http://services.xmethods.net/soap/urn:xmethods-
delayed-quotes.wsdl");
// create service
javax.xml.rpc.Service service =
factory.createService(wsdlLocation, serviceName);
// create call
javax.xml.rpc.Call call =
service.createCall(portName, operationName);
//Populate an array with list of stock names
Vector populatedStockNames = this.getPopulatedList();
//Loop through the populated list, and for each stock get a result
java.util.Iterator i = this.getPopulatedList().iterator();
String stockName = null;
Float result;
com.ddj.wsstruts.valueobject.StockValue stockValue = null;
while(i.hasNext()) {
stockName = (String) i.next();
// invoke the remote web service
result = (Float) call.invoke(new Object[] {stockName});
if(category.isDebugEnabled()) {
 category.debug(" The quote for " + stockName + " is: " + result);
}
//Set stock name and stock values in stockValue bean
stockValue = new com.ddj.wsstruts.valueobject.StockValue();
stockValue.setStockName(stockName);
stockValue.setStockValue(result.toString());
//Add the stockValue bean to stockValues vector
stockValues.add(stockValue);
}
}

一旦 Action 類調用完模型服務和 WSManager, ActionForm中的股票值及信息內容域就會被填充。控制就被返回到 JSP頁面,從 ActionForm 獲取所有必需的值並打印 UI.

結束語

在本文中,您看到了 Struts架構是如何與 Web 服務相集成的。您也了解了如何使用 Struts組件來提供和預訂 Web服務。本文附帶的簡單應用程序代碼將幫助您深入理解所有這些是如何工作的。

使用此處所闡明的體系結構,您可以開發出這樣的企業應用程序,它非常健壯,很容易維護,而且能很容易地與早期應用程序集成在一起。我希望您將可以開發出更多的關於 Struts 和 Web服務的項目,並看到這種體系結構在您自己的項目中是多麼的有用。

參考資料

  • 下載本文附帶的 zip文件,以開始使用這裡所闡述的體系結構。
  • project 的主頁查找更多關於 Struts 框架的信息(由 Apache項目託管)。

  • 查看 Malcolm Davis所寫的「Struts,MVC 的一種開放源碼實現」(developerWorks,2001年2月)。雖然這個項目的觀點有點過時了,但對於項目目標與一般的 MVC模式仍然是一個很好的入門介紹。

  • 閱讀 Wellie Chao 的「Struts 和 Tiles 輔助基於組件的開發」(developerWorks,2002年6月),以了解更多關於 Struts的信息。

  • 查看 BrettMcLaughlin 的「實體 bean 保護」(developerWorks,2002年10月),以了解更多關於 Session Fac,ade模式的信息。

  • XMethods是代理服務的一個示例。

冷日
(冷日)
Webmaster
  • 註冊日: 2008/2/19
  • 來自:
  • 發表數: 15771
[分享]為您的 Linux 應用開發加油提速 2003:創建基於 Struts 的 Web 應用程序
使用 IBM WebSphere Studio 開始學習 Struts 的提示和技巧

本文演示了如何使用 IBM 的 WebSphere Studio 產品來創建一個使用 Struts 的簡單 Web 應用程序,Struts
是一個用於開發 Web 應用程序的概念框架,它是 Apache 軟件基金會 Jakarta
項目的一部分。如果您有任何疑問、意見,或者希望尋求幫助,請訪問我們的技術支持論壇

簡介
Struts 是一個用於開發 Web 應用程序的概念框架。它是模型-視圖-控制器(Model-View-Controller,MVC)設計範例的一個變體,其中控製器的功能是作為應用程序邏輯(模型)和數據表示(視圖)之間的中間介質。該框架包含一組 Java 類和 JSP 標記庫。同 IBM WebSphere Studio 中的工具一起,Struts 可以幫助開發人員快速地設計和實現 Web 應用程序。Struts 是 Apache 軟件基金會 Jakarta 項目所擁有的眾多產品中的一種。

本文為您演示了如何使用 IBM WebSphere Studio Site Developer 來創建一些初始組件,它們是基於 Struts 的 Web 應用程序所必需的。這些組件包括:

  • 應用程序的網站的入口頁面(index.jsp),它包括用於輸入用戶標識和密碼的登錄表單
  • 用於初始數據驗證的相關 ActionForm 類,它驗證用戶是否輸入了用戶標識和密碼,如果未輸入則發出錯誤消息
  • 認證用戶的用戶標識和密碼的 Action 類(在實際的應用程序中,應當由 WebSphere 或 IBM Policy Director 管理安全性,例如用戶標識/密碼認證)
  • 網站的 home.jsp,一旦成功登錄就調用它
  • error.jsp 頁面,如果輸入的用戶標識和密碼未能通過認證,則顯示該頁面

圖 1. 樣本 Struts 應用程序
樣本 Struts 應用程序

開始之前
請安裝 Red
Hat Linux 7.2
、Red Hat Linux 7.3、SuSE Linux 7.3安裝 SuSE Linux Enterprise Server(SLES)8


將需要 WebSphere Studio Site Developer。通過索取包含 WebSphere Studio Site
Developer 和其它 IBM 試用軟件的免費 CD 集,或者通過下載,可以獲得免費的 60 天試用版本。可在「Speed-start
your Linux application」概述頁面上找到有關索取 CD 集或下載產品
信息。儘管本文中的示例使用了 IBM WebSphere Studio Site Developer,但是也可以使用 IBM
WebSphere Studio Application Developer,它包含了其它特性,例如 EJB
支持。本文以下部分將這兩個產品都稱為「WebSphere Studio」。

您應當使用 Help->Software Updates 菜單功能將您的 WebSphere Studio 軟件升級到 5.0.1 級別。這將糾正下面提及的一些接口問題;否則,這些問題將要求您停止和重新啟動 WebSphere Studio。

您是 Linux 新手嗎?如果您是一位 Linux 新手,並且需要有關登錄和註銷、掛裝 CD-ROM 或打開終端窗口方面的幫助,請參閱「針對 Linux 開發新手的基本任務」,它介紹了這些任務,以及其它任務。

創建 Web 項目
在本節中,我們將登錄,然後啟動 WebSphere Studio 並在新的工作空間中創建用於 Struts 應用程序的 Web 項目。

  1. 以某個用戶身份(不要以 root 用戶身份)登錄並啟動 WebSphere Studio Site Developer。從終端窗口運行wssitedev50
    如果您的路徑中無 /usr/bin,那麼請運行 /usr/bin/wssitedev50。還可以從桌面的任務欄啟動程序。例如,對於 KDE,選擇 Start Application > Run Command... 並根據情況相應地輸入 wssitedev50/usr/bin/wssitedev50。對於本示例,我們將創建一個名為 MyBank 的新工作空間,以便於將該項目與您可能正在進行的其它項目區分開。

    圖 2. 啟動新項目
    啟動新項目

  2. 創建工作空間時,您會看到顯示 Site Developer 徽標。這一工作可能需要花費一些時間,請耐心等待。

    圖 3. 創建 MyBank 工作空間
    創建 MyBank 工作空間

  3. 當工作空間打開時,您的首要任務是創建一個新項目。對於本項目,我們將向您介紹基於 Struts 的 Web 應用程序備忘單(cheat sheet)。有關創建項目的更多方法,請參閱文章「用 WebSphere Studio Application Developer 在 90 秒內快速啟動您的 Linux 應用開發:「Hello, World」」。順帶我們將指出備忘單的功能,並且將介紹一些備忘單未涉及的一些功能。要啟動一個帶有備忘單的新 Struts 項目,請單擊 Help > Cheat Sheets > Create a Struts-based Web application,如圖 4 所示。

    圖 4. 創建新項目
    創建新項目

  4. 打開了備忘單介紹。現在,在左邊的導航區域有一個備忘單圖標,當看不見備忘單時,可以使用該圖標來打開它。還有一個形狀為黑色箭頭的按鈕,可以使用它跳到備忘單的下一步。現在單擊它以轉到備忘單的下一個階段,在那將開始創建我們的項目。

    圖 5. 備忘單介紹
    備忘單介紹

  5. 單擊執行按鈕(黑色箭頭)以創建一個 Web 項目。

    圖 6. 備忘單 — 創建 Web 項目
    備忘單 — 創建 Web 項目

  6. 在下一個屏幕上,在 Project name 域中輸入 MyBankWeb。備忘單已經選擇了 Web Project features 下的 Add Struts support。選項 Create a default CSS file 應當已處於被選中狀態。單擊 Next

    圖 7. 創建新的 Web 項目
    創建新的 Web 項目(1)

  7. 在 J2EE Settings Page 上,在 New project name 域中輸入 MyBankEAR,並在 Context root 域中輸入 MyBank。使用缺省的 J2EE 級別 1.3。單擊 Next

    圖 8. 指定 J2EE 設置
    指定 J2EE 設置

  8. Struts Settings 窗口上的缺省值是可接受的。請注意選中了Create a Resource Bundle for Struts Project。在下一部分處理中,我們將向該資源束(Resource bundle)添加屏幕文本和出錯消息特性。在資源束中保存這樣的字符串有助於將我們的 Web 頁面翻譯成另一種語言。單擊 Finish

    圖 9. 指定 Struts 設置
    指定 Struts 設置

  9. 您將返回到備忘單,現在多了一個處於選中狀態的項。單擊備忘單上的最小化(_)按鈕,因為我們在下一步中要使用工作空間的其它一些特性。

    圖 10. 備忘單 — 創建 Web 圖
    備忘單 — 創建 Web 圖

WebSphere Studio 中的透視圖

繼續之前,我們需要介紹一個重要的概念。WebSphere Studio 是一個面向任務的工具,它的用戶界面是為要執行的任務定製的。用於開發
JSP 的界面與用於服務器開發的界面有很大的不同。通過切換您正在使用的透視圖,Site Developer 允許您對查看哪些窗口和視圖進行控制。透視圖是一組視圖,這些視圖顯示了執行具體任務時您需要的資源。WebSphere Studio 擁有為 Web 應用程序開發、J2EE 開發、調試和許多其它常見任務而定義的透視圖。左邊的導航欄包含了幾個圖標,這些圖標允許您方便地打開透視圖或在透視圖之間切換。

打開透視圖圖標
打開透視圖
打開新的透視圖或切換到一個已經打開的透視圖。常用的透視圖是列在菜單中的,其它許多透視圖可通過 Other 菜單選項訪問。
Web 透視圖圖標
Web 透視圖
將視圖切換到 Web 透視圖。
備忘單圖標
備忘單
顯示或最小化備忘單

應用程序資源

面我們提到過,我們將創建資源束,以允許翻譯文本。根據這個應用程序的初始設計,它將包含三個 JavaServer
Pages:index.jsp、home.jsp 和
error.jsp。我們將為您演示如何為要在所有頁面、出錯消息、按鈕文本和用戶標識及密碼提示上使用的標題創建應用程序資源。儘管我們在實際的應用程
序中不會混合使用不同的方法,但是我們還是會將一些文本直接編輯進頁面,這樣您就可以看到不同的工作方法。

這些資源存儲在應用程序資源特性文件中。在本節中,我們將為您演示幾個有關 IDE 的技巧,然後創建這些應用程序資源。

  1. 如果你從頭至尾都是按照我們的步驟進行的,那麼您現在應當看到了 Web 透視圖。如果沒有,那麼請單擊左邊導航欄中的Web 透視圖圖標(見上面的 WebSphere Studio 中的透視圖中的描述),或者選擇 Window > Open Perspective > Web。我們將使用 Web 透視圖中的 J2EE Navigator 窗格來瀏覽應用程序資源文件,然後打開它進行編輯。窗格底部的選項卡允許您在窗格之間切換。

    圖 11. J2EE Navigator 窗格
    J2EE Navigator 窗格

  2. 現在通過單擊加號(+)展開 Web 項目樹,以顯示如圖 12 所示的ApplicationResources.properties 文件。如果 J2EE 窗格非常小,可以在標題上按鼠標右鍵以彈出上下文菜單,然後最大化該窗格。

    圖 12. 展開的 J2EE Navigator 窗格
    展開的 J2EE Navigator 窗格

  3. 雙擊 MyBankWeb.Web Content.WEB-INF.classes.mybankweb.resources 中的 ApplicationResources.properties,以利用特性文件編輯器(Properties File Editor)打開它。請注意當您這樣做時 J2EE Navigator 窗格是如何恢復成以前大小的。儘管超出了本文的範圍,但是如果您願意,也可以使用其它編輯器。選擇 Window > Preferences,並且在首選項樹的 Workbench 部分的 File associations 下面進行查看。

    除去 errors.header 和 errors.footer 特性前的 #。在 errors.footer 特性之後添加如下面清單所示的那些行。

    提示:Linux 的應用程序有時候會共享剪貼板,但並非始終如此。在共享的情況下,Edit >
    copy(ctrl-C)和 Edit >
    paste(ctrl-V)通常分別用做複製和粘貼。如果這無法工作,那麼如果您有一個三鍵鼠標(或者如果您已經對鼠標進行了設置,通過協調兩個按鍵

    即同時按下兩個按鍵來模擬三鍵鼠標),通常有一個較為「古老」的方法可以使用。按下鼠標左鍵,在某個窗口的文本上拖動鼠標,然後使用中間的按鍵(或在
    2 鍵鼠標上同時按下兩個按鍵)將突出顯示的文本粘貼進編輯器窗格。您可以在一個窗口中或多個窗口之間使用這個便捷的技巧。

    清單 1. 將這些行添加到 ApplicationResources.properties 文件中的 errors.footer 特性之後

    # Optional header and footer for <errors/> tag.
    errors.header=<ul>
    errors.footer=</ul>
    index.title=MyBank Application
    button.login=Login
    global.field.userid=User ID
    global.field.password=Password
    error.login.nouserid=<li>You must enter a User ID.
    error.login.nopassword=<li>You must enter a Password.
    error.login.failed=<li>Invalid User ID and/or Password entered.
    error.login.exception=<li>Exception occurred in action.


  4. 剛剛添加的值將表示 JSP 文件中的顯示輸出。請注意,我們的出錯消息是以無序列表中的列表元素的形式顯示的。保存並關閉
    ApplicationResources.properties 文件。在缺省的編輯器中,如果您願意,可以使用鍵盤快捷鍵:Ctrl-s
    將保存文件,Ctrl-F4 將關閉該窗格。

Struts 應用程序圖
在這一步中,我們將返回到備忘單,並且創建 Struts 應用程序圖,我們將用它來設計和構建我們的應用程序組件。

  1. 單擊左邊導航欄中的備忘單圖標以將其恢復成圖 10 中的備忘單。單擊 Perform 按鈕創建新的 Web 圖。

    如果您沒在使用備忘單,那麼通過單擊工具欄按鈕(New 嚮導按鈕)(或選擇 File > New > Other.... )可以打開 New 嚮導。在 Select 對話框的左邊窗格上,展開 Web 並選擇 Struts。在右邊的窗格中選擇 Web Diagram。單擊 Next

    註:如果您不喜歡使用 Struts 應用程序圖,那麼通過使用 File > New > Other... 下可用的新建文件嚮導,然後選擇Web > Struts 和您希望構建的組件,您仍可以為該項目創建每個組件。

  2. 在 File name 域中輸入 Login。單擊 Finish

    圖 13. 創建新的 Web 圖
    創建新的 Web 圖



    Struts 應用程序圖文件 Login.gpf 是用 SADE(Struts
    應用程序圖表編輯器)創建和打開的。現在將在自由窗體界面(Free Form Surface,FFS)上布置您的 JavaServer
    Pages、表單 Bean 和 Action
    Bean。我們將使用下列位於編輯器上方的工具欄按鈕。(除非您將鼠標移動到這些按鈕上方,否則它們是灰色的,不可點擊。)

    連接圖標
    連接(Connection)
    繪製從選定節點到目標節點的連接
    操作映射圖標
    操作映射節點(Action Mapping Node)
    使光標準備就緒以將新的操作映射放到 FFS 上
    表單 bean 圖標
    表單 Bean 節點
    使光標準備就緒以將新的表單 bean 節點放到 FFS 上
    java bean 圖標
    Java Bean 節點
    使光標準備就緒以將新的 java bean 節點放到 FFS 上
    Web 頁面圖標
    Web 頁面節點
    使光標準備就緒以將新的 Web 頁面放到 FFS 上

  3. 我們現在將創建 Web 頁面對象。首先單擊工具欄中的「新建 Web 頁面節點(New Web Page Node)」按鈕(Web 頁面圖標)(或者在圖表編輯器界面中單擊鼠標右鍵,然後選擇 New > Web Page Node),然後在圖表編輯器的左邊單擊以放下 Web 頁面節點。將該頁面的路徑從缺省的 /page.jsp 更改為 /index.jsp,然後按 Enter。在新的 JSP 上單擊鼠標右鍵以彈出上下文菜單,然後選擇 Change description。輸入 Login entry page 的描述,並按 Enter。您的 FFS 現在應當看起來如圖 14 所示。如果弄錯了路徑或描述,可以使用上下文菜單更改它們。

    圖 14. 創建新的 JSP
    創建新的 JSP

  4. 以同樣的方式,在 FFS 上創建另兩個新的 Web 頁面節點,並將它們的路徑名分別更改為 /home.jsp/error.jsp,如圖 15 所示。如果願意,可以添加描述。

    圖 15. 創建其它 JSP
    創建其它 JSP

  5. 以類似的方式,將操作映射節點(操作映射圖標)放在 FFS 的中間,並將其路徑設置為 /login。然後創建表單 Bean 對象(表單 Bean 圖標)。在 Form Bean Attributes 中,在 Name 域中輸入 loginForm,並保留 Scope 為 request。單擊 OK

    圖 16. 創建表單 bean
    創建表單 bean

  6. 現在我們將創建節點之間的連接。Struts 圖連接是一條線,它表示了兩個 Struts 節點之間的邏輯流或數據流。要創建連接,請單擊工具欄中的「連接兩個節點(Connect Two Nodes)」按鈕(連接圖標)。首先單擊源對象,再單擊目標對象,以在編輯器中創建兩個節點之間的連接。
    1. index.jsp(登錄入口)頁面連接到 login 操作映射。
    2. login 操作映射連接到 home.jsp 頁面。將連接 的轉發名(forward name)替換為 success,然後按 Enter。如果稍後需要更改連接的轉發名,只需在連接線上單擊鼠標右鍵,然後選擇Edit the forward name
    3. login 操作映射連接到 error.jsp 頁面。將連接 的轉發名替換為 failure,然後按 Enter
    4. error.jsp 頁面連接到 index.jsp(登錄入口)頁面。
    5. index.jsp(登錄入口)頁面連接到 loginForm Bean。


    建立了這些連接後,應當擁有一個類似圖 17 的布局。

    圖 17. 添加連接
    添加連接


    請注意,對像是灰色的,連接是用虛線表示的,因為它們尚未實現。

實現 JavaServer Pages
現在我們將返回到備忘單,以實現我們的 JSP:index.jsp(登錄入口頁面)、主頁和出錯頁面。

  1. 再次顯示備忘單,並在 Create the JSPs (WebPages) and Beans 部分中單擊 Perform 按鈕。在產生的窗口上,展開 MyBankWeb 並單擊一次以選擇 Login 表單,然後單擊 OK。將打開一個新窗口,其中帶有一個未實現的 JSP 和表單 bean 的檢查表。我們將從 index.jsp 著手,所以選擇它並單擊 OK(或者雙擊它)。

    圖 18. 未實現的 JSP 和 bean 的列表
    未實現的 JSP 和 bean 的列表

  2. 現在我們將使用頁面設計器(Page Designer)來創建 index.jsp。
    1. 將打開新建 JSP 嚮導。如果您並未在使用備忘單,也可以打開該嚮導,即通過在應用程序圖(Login.gph)中的 index.jsp 節點上雙擊。File Name 已被設置為 index,而 Model 被設置為 Struts JSP。單擊 Finish

      圖 19. 新建 JSP 嚮導
      新建 JSP 嚮導

    2. 頁面設計器將以設計方式(Design mode)啟動。在開始編輯我們的 JSP 之前,我們將為您介紹 Studio IDE 中的一些便捷的視圖。從 IDE 左下方窗格中的選項卡(Gallery、Library 和 Outline ……)單擊 Outline 選項卡以選擇 outline 視圖,這將向我們提供樹結構的 JSP outline 視圖,該視圖將有助於我們確保我們的頁面結構是正確的。

      右下窗格包含一條表明 /login 目標不存在的消息。我們需要編輯幾個特性,所以這裡我們將打開 properties 視圖。要做到這一點,請選擇Window > Show View > Other。在彈出的對話框中,展開 Basic,選擇Properties,然後單擊 OKProperties 選項卡將被添加到右下窗格,該窗格將在 Properties 視圖中打開。您的屏幕現在應當如圖 20 所示。

      圖 20. Outline 和 Properties 視圖
      Outline 和 Properties 視圖

    3. 現在我們回過頭創建 JSP。應當已經存在了一個 標記集(由一個邊框所表示的),其操作被設置為 /login,以調用登錄操作映射。在邊框中單擊以選擇它。當它被選中時您會看到框的輪廓發生變化。還應在 Outline 視圖中選擇 html:form 元素。一旦選擇了它,單擊 JSP 菜單並選擇 Insert Custom

      圖 21. 頁面設計
      頁面設計

    4. 如圖 22 所示選擇 beanmessage。單擊 Insert。選擇 htmltext,並再次單擊 Insert。然後單擊 Close。這將插入一個用於用戶標識提示的消息 bean,以及一個要求用戶輸入標識的文本域。請注意 Outline 視圖中的其它項。

      圖 22. 添加消息 bean 和文本域
      添加消息 bean 和文本域

    5. 在上一個步驟中插入到登錄表單的消息 bean 定製標記應當有表明 Missing message for key

      的文本。單擊該 bean:message 定製標記以選擇它(或者在 Outline 視圖中選擇它)。這將在 Properties 視圖中顯示
      bean 的特性。如果 Properties 視圖非常小,您可能希望把它放大,這可以象圖 23
      中所做的那樣,通過使用鼠標左鍵拖動它的上部將其往上拉來做到這一點。向下滾動 Properties 視圖,然後將消息的 key 特性更改為 global.field.userid。按 Enter。這將在應用程序資源文件中使用與 global.field.userid 相關的文本。用同樣的方法,選擇您插入的 html:text 定製標記,並將其 property 特性更改為 userid。這將是用戶標識變量在 Java 代碼中的名稱。當您更改特性時,可能會有一些延遲,這取決於您的處理器速度和所擁有的 RAM 數量。

      圖 23. 設置消息 bean 的 key 特性
      設置消息 bean 的 key 特性

    6. 將光標放到 html:text 域之後,然後按 Enter 插入一個換行符(BR)。現在我們將插入其它兩個定製元素,以創建我們的密碼提示和輸入域,所以請單擊 JSP 菜單然後選擇 Insert Custom。選擇 beanmessage,並單擊 Insert。選擇 htmlpassword,然後單擊 Insert 以添加密碼輸入域。單擊 Close。將新的消息 bean 的 key 特性設置為 global.field.password,並單擊Enter。將 property 特性設置為 password,然後單擊 Enter。您的屏幕看上去應當象圖 24。

      圖 24. 添加密碼提示和輸入域
      添加密碼提示和輸入域

    7. 既然已經定義了數據輸入域,我們需要一個按鈕來提交該表單。將光標放在 html:password 域之後並按兩次 Enter,以插入兩個換行符。現在使用 JSP 菜單並選擇 Insert Custom。現在選擇 htmlSubmit 並單擊 Insert,然後選擇 beanmessage,再次單擊 Insert。單擊Close。在 Properties 視圖中,輸入 button.login 作為新消息 bean 的 key 特性,並按 Enter。(如果您尚未運行 Help > Software updates 升級到 V5.0.1,那麼 button.login 文本可能不會立即出現在 Design 視圖中)。如果一切正常,您的屏幕現在看起來應該如圖 25。

      圖 25. 添加提交按鈕
      添加提交按鈕

    8. 在表單上方(在它外面)單擊鼠標左鍵以添加標題標記。您的光標應該正好在表單前面。Outline 視圖應當突出顯示「BODY」。從菜單欄選擇 Format > Paragraph > Heading 1。通過插入另一個 bean:message JSP 定製標記來添加 Login 頁面標題文本。將 key 特性設置為 index.title 並按 Enter。應當出現 MyBank Application 標題。

      圖 26. 添加頁面標題
      添加頁面標題

    9. 在標題和表單之間的位置上單擊鼠標左鍵。然後插入另一個 JSP 定製標記html:errors

      圖 27. 添加 html:errors 標記
      添加 html:errors 標記

    10. 通過單擊 Source 選項卡可以查看我們頁面的源文件。Login 入口頁面的 Source 視圖如圖 28 所示。請注意 action="/login" 旁邊有一個警告,因為我們尚未為其定義操作映射。

      圖 28. Source 視圖
      Source 視圖

    11. 使用 File > Save index.jsp 保存 JSP,然後使用 File > Close 關閉它。請注意應用程序圖中的 index.jsp 節點現在加上了色彩,表明它已經實現了。在備忘單的 JSP 和 bean 列表中還有一個選中標記。
  3. 既然創建了 Login 入口頁面(index.jsp),現在可以創建網站的 home.jsp 頁面。
    1. 象上一步對 index.jsp 節點所做的那樣,在備忘單的 JSP 和 bean 列表中選擇 home.jsp 項,然後單擊 OK;或者在應用程序圖(Login.gph)中雙擊 home.jsp 節點。這將啟動 New JSP File 嚮導。File Name 已被設置為 home.jsp,而 Model 被設置為 Struts JSP。單擊 Finish
    2. 切換到頁面設計器的 Source 視圖,並且刪除文本

      Place home.jsp's content here.

      。用下面這些行代替它以實現主頁:(為了節省時間您可以複製/粘貼下面的代碼。)
      <h1><bean:message key="index.title"/></h1>
      
      <h2>Welcome <bean:write name="loginForm"  property="userid"/></h2>
      <h2>To MyBank Home Page</h2>

      請注意,主頁正在使用 loginForm 的用戶標識 bean 特性來輸出用戶標識的輸入。考慮將如何在 Design 視圖中添加 bean:write。還請注意 H2 元素並未使用來自應用程序資源文件的文本,而是在使用靜態文本。正如我們在前面所說的,我們在此說明兩種方法,但是像這樣混合使用文本源不是好的做法。
    3. 切換到 Design 視圖以查看該頁面的外觀。然後保存並關閉 home.jsp。
  4. 現在已經創建了 index.jsp 和 home.jsp 頁面,您可以創建 error.jsp 頁面了。
    1. 在應用程序圖(Login.gph)中,就像您在前面步驟中對 index.jsp 節點所做的那樣雙擊 error.jsp 節點。這將啟動 New JSP File 嚮導。Model 和 File Name 已經設置好了。單擊 Finish
    2. 在頁面設計器的 Source 視圖中,刪除文本

      Place home.jsp's content here.

      。利用下面這些行替代它以實現出錯頁面:(為了節省時間您可以複製/粘貼下面的代碼。)
      <h1><bean:message key="index.title"/></h1>
      
      <h2>Error Page</h2>
      <p>
      <html:errors/>
      <p>
      <a href="index.jsp"><b>Return to Login Page</b></a>
    3. 切換到 Design 視圖以查看該頁面的外觀。然後保存並關閉 error.jsp。
  5. 現在,應用程序圖上的所有三個 JSP 頁面都應當添加了色彩。

    圖 29. 帶有已實現的 JSP 的應用程序圖
    帶有已實現的 JSP 的應用程序圖

實現 ActionForm 類

  1. 現在需要實現 loginForm 類以處理來自 index.jsp 的輸入。這個 ActionForm 類將包含一個字段,以及用於輸入表單中每個域的取值方法(getter 和 setter),並且將提供簡單的域檢查。要幫助實現該類,您將使用 New ActionForm Class 嚮導。創建 ActionForm 類之後,然後您將把代碼添加到該類中的 validate 方法。
    1. 在應用程序圖(Login.gph)中,雙擊 loginForm 節點。這將啟動 New ActionForm Class 嚮導。ActionForm 類名已經被設置為 LoginForm。單擊 Next如果 Next 按鈕被任務欄遮蔽了,那麼您可能必須最小化任務欄,或者將它拖到屏幕的另一邊。

      圖 30. 啟動 ActionForm 嚮導
      啟動 ActionForm 嚮導

    2. 在「Choose new accessors for your ActionForm class」上,展開 MyBankWeb、Web Pages、index.jsp 和 /login。對於 Choose Accessors,請選擇 text:useridpassword:password。單擊 Next。(註:如果您未看到 text:userid 和 password:password,您可能需要關閉 WebSphere Studio,然後重新啟動它。)

      圖 31. 選擇取值方法
      選擇取值方法

    3. 下一個頁面顯示了已選擇的新取值方法(New Accessors)。單擊 Next
    4. 下一個頁面為您的 ActionForm 類創建映射。請檢查 Configuration File Name 為 WEB-INF/struts-config.xml,而 Mapping Name 為loginForm。單擊 Finish

      圖 32. 配置文件名和映射名
      配置文件名和映射名

    5. Java 編輯器現在將打開 LoginForm.java。請注意,該嚮導已為您創建了 ActionForm 類的大部分內容,包括兩個字段(userid 和 password)及其 setter 和 getter 方法。向下滾動文件至 ActionErrors validate 方法(接近源文件的底部),用以下代碼替代該方法。
      public ActionErrors validate(
      ActionMapping mapping,
      HttpServletRequest request) {
      
      ActionErrors errors = new ActionErrors();
      // Validate the fields in your form, 
      // adding each error to this.errors as found
       if ((getUserid() == null) || (getUserid().length() == 0))
      errors.add("userid", 
      new ActionError("error.login.nouserid"));
      if ((getPassword() == null) || (getPassword().length() == 0))
      errors.add("password", 
      new ActionError("error.login.nopassword"));
      return errors;
      } 
    6. 保存並關閉 LoginForm.java。(如果剛才必須最小化任務欄或將其移到不礙事的地方,那麼現在您可以恢復它。)
  2. 現在,您可以檢查 Struts 配置文件,也可以不檢查。在 J2EE Navigator 視圖中,雙擊 Web Content/WEB-INF/ 目錄中的 struts-config.xml,以便用 Struts 配置文件編輯器(Struts Configuration File Editor)打開它。選擇 Form Beans 選項卡並選擇 Bean Name 中的 loginForm。檢查配置文件之後關閉它。

    圖 33. 檢查 Struts 配置文件
    檢查 Struts 配置文件

  3. 關閉備忘單的 JSP 和 bean 列表。

實現操作映射(Action Mapping)和操作類(Action Class)
實現的最後一部分是實現登錄操作映射和 LoginAction 類。將用來自登錄表單的數據調用 LoginAction 類。該類可用於實現對業務邏輯的調用,就像包裝器類一樣。它還可用於確保調用業務邏輯的狀態和條件是正確的。在本練習中,通過檢查用戶標識和密碼,用它來驗證用戶的權限。


驗證用戶標識和密碼,通常將 LTPA 和權限 LDAP 目錄(例如 IBM Directory)或者其它一些權限管理支持(例如 IBM
Policy Director)一起使用。在本示例中,將只根據固定的用戶標識和密碼(userid/password)檢查用戶標識和密碼。

  1. 顯示備忘單。您應當正在查看 Create the actions 部分。單擊 Perform 按鈕以打開未實現操作的列表。
    1. 就像在上面對 JSP 和 bean 所做的那樣,選擇未實現的操作並單擊 OK。這將打開 New Action Mapping 嚮導以創建我們的操作映射和操作類。如果您並未在使用備忘單,那麼雙擊應用程序圖(Login.gph)中的 login 操作映射節點。這樣會啟動 New Action Mapping 嚮導。Configuration File Name 已經被設置為 WEB-INF/struts-config.xml,Mapping Path 被設置為 /login,Forwards 中,Name 被設置為您定義的名稱(successfailure)。從 Form Bean Name 下拉菜單選擇 loginForm。從 Form Bean Scope 下拉菜單選擇 request。單擊 Next

      圖 34. New Action Mapping 嚮導
      New Action Mapping 嚮導

    2. 在「Create an Action class for your mapping」上,Action Class Name 已經被設置為 LoginAction。單擊 Finish。我們不再需要備忘單的操作列表了,所以在它上面單擊 Close
    3. Java 編輯器將打開 LoginAction 類文件。向下滾動到 perform 方法,並利用下列代碼替換它。
      public ActionForward perform(
        ActionMapping mapping,
        ActionForm form,
        HttpServletRequest request,
        HttpServletResponse response)
        throws IOException, ServletException {
        ActionErrors errors = new ActionErrors();
        ActionForward forward = new ActionForward();
        // return value 
        LoginForm loginForm = (LoginForm) form;
        try {
          // do something here
          if (!loginForm.getUserid().equals("userid")
            || !loginForm.getPassword().equals("password"))
            errors.add("login", new ActionError("error.login.failed"));
        } catch (Exception e) {
          // Report the error using the appropriate name and ID.
          errors.add("login", new ActionError("error.login.exception"));
        }
        // If a message is required, save the specified key(s)
        // into the request for use by the <struts:errors> tag.
        if (!errors.empty()) { 
          saveErrors(request, errors);
          // Forward control to the failure URI
          forward = mapping.findForward("failure");
        } else {
          // Forward control to the success URI
          forward = mapping.findForward("success");
        }
        // Finish with
        return (forward);
      }
    4. 保存並關閉 LoginAction.java。既然已實現了所有組件,那麼 Login.gph 上的組件現在都是有顏色的了。

      圖 35. 應用程序圖中的所有組件都已實現
      應用程序圖中的所有組件都已實現

  2. 既然已實現了 LoginAction 類,您可以檢查 Struts 配置文件以查看圖表編輯器(Diagram Editor)和嚮導做了哪些更改。
    1. 在 J2EE Navigator 視圖中,雙擊 Web Content/WEB-INF/ 目錄中的 struts-config.xml 以在 Struts 配置文件編輯器(Struts Configuration File Editor)中打開它。
    2. 選擇 Actions 選項卡。然後在 Action Path 中選擇 /login。如果尚未設置好,那麼在 Input 域中輸入 /index.jsp。Specify
      Forwards 的域應當由圖 34 中 Forwards 域的 Name 中為 JSP 頁面指定的名稱填充,這些名稱分別與對應的 JSP
      頁面相關(success 與 /home.jsp 相關,而 failure 與 /error.jsp 相關)。

      圖 36. 設置操作輸入
      設置操作輸入

    3. 現在,通過選擇 XML Source 選項卡來檢查 struts-config.xml 文件的 XML 源代碼。
    4. 保存和關閉 struts-config.xml 文件。

測試 Struts 應用程序
既然已經完成了 Struts 應用程序,那麼可以利用內置的 WebSphere V5.0 測試環境(Test Environment)測試它。

  1. 再次顯示備忘單。您應該正好看到 Run the Web Application 部分。單擊執行按鈕(黑色箭頭)。如果您沒有使用備忘單,那麼可以在 J2EE Navigator 窗格中選擇 index.jsp。然後用鼠標右鍵單擊 index.jsp,並選擇 Run on Server ...
  2. 在 Server selection 對話框中,選擇 WebSphere V5.0 Test Environment 並單擊 OK

    圖 37. 設置操作輸入
    設置操作輸入

  3. 該項目將被發布到測試環境服務器上。將啟動服務器(Server),並且打開控制台(Console)視圖。當服務器準備就緒時,控制台將顯示一條以「Server server1 open for e-business」結尾的消息,該消息後面跟著另一條以 /index.jsp: init 結尾的消息。

    圖 38. 服務器測試環境啟動
    服務器測試環境啟動



    註:現在如果您關閉已完成的備忘單,那麼查看控制台將更容易。
  4. 然後 index.jsp 將在 Web 瀏覽器中打開。輸入用戶標識 userid 和密碼 password。單擊 Login

    圖 39. 登錄
    登錄

  5. 稍候片刻,應用程序將顯示 .jsp 頁面:

    圖 40. MyBank 主頁
    MyBank 主頁

  6. 單擊瀏覽器的 Back 按鈕,並嘗試輸入不同的用戶標識或密碼,以查看出錯頁面。單擊 Login

    圖 41. 出錯頁面
    出錯頁面

  7. 然後將在 Web 瀏覽器中打開 index.jsp。輸入用戶標識 userid 和密碼 password。單擊 Login

    圖 42. 未輸入用戶標識或密碼
    未輸入用戶標識或密碼

結束語

喜您!您剛才已經創建了一個基於 Struts 的 Web 應用程序。您使用 WebSphere Studio 構建了 JavaServer
Pages、ActionForm、操作類和操作映射。您使用 Struts 應用程序圖表編輯器(Struts Application
Diagram Editor)設計和創建了您的應用程序。然後您利用 WebSphere Studio V5.0 測試環境(它包含在
WebSphere Studio 中)測試了您的應用程序。如果您有任何問題,請訪問我們的技術支持論壇

致謝
非常感謝 Ian Shields 為 Struts 備忘單補充了材料,並增加了有關 WebSphere Studio IDE 的使用技巧,他還通過大量的測試進行了其它許多修改。

參考資料

前一個主題 | 下一個主題 | 頁首 | | |



Powered by XOOPS 2.0 © 2001-2008 The XOOPS Project|