當(dāng)前位置:首頁 > 芯聞號 > 充電吧
[導(dǎo)讀]前一段時(shí)間好象有人問關(guān)于Struts的事情,本人留意了一下.請有興趣的同志不防研究研究. (1) 1. 介紹 1.1 Model-View-Controller (MVC) 設(shè)計(jì)模式 FIXME - 

前一段時(shí)間好象有人問關(guān)于Struts的事情,本人留意了一下.請有興趣的同志不防研究研究.
(1)
1. 介紹

1.1 Model-View-Controller (MVC) 設(shè)計(jì)模式

FIXME - 需要一個(gè)對該模式一般性的介紹。(譯注:可以參考機(jī)械工業(yè)出版社的《設(shè)計(jì)模式》。)
 
1.2 將MVC概念映射到Struts組件中

Struts 的體系結(jié)構(gòu)實(shí)現(xiàn)了Model-View-Controller設(shè)計(jì)模式的概念,它將這些概念映射到web應(yīng)用程序的組件和概念中?!?
這一體系結(jié)構(gòu)中每個(gè)主要的組件都將在下面做詳細(xì)的討論?!?

1.3 Model: 系統(tǒng)狀態(tài)和商業(yè)邏輯JavaBeans

基于MVC的系統(tǒng)中的 Model 部分可以細(xì)分為兩個(gè)概念 -- 系統(tǒng)的內(nèi)部狀態(tài), 能夠改變狀態(tài)的行為。用語法術(shù)語來說,我們可以把狀態(tài)信息當(dāng)作名詞(事物),把行為當(dāng)作動(dòng)詞(事物狀態(tài)的改變)?!?
通常說來,你的應(yīng)用程序?qū)⑾到y(tǒng)內(nèi)部的狀態(tài)表示為一組一個(gè)或多個(gè)的JavaBeans,使用屬性(properties)來表示狀態(tài)的細(xì)節(jié)。依賴于你的應(yīng)用程序的復(fù)雜度,這些beans可以是自包含的(以某種方式知道怎樣永久地保存它們的狀態(tài)信息),或者可以是正面的(facades),知道當(dāng)被請求時(shí)怎樣從外部數(shù)據(jù)源(例如數(shù)據(jù)庫)中取得信息。Entity EJBs通常也用來表示內(nèi)部狀態(tài)。
大型應(yīng)用程序經(jīng)常將系統(tǒng)可能的商業(yè)邏輯行為表示為可以被維護(hù)狀態(tài)信息的beans調(diào)用的方法。舉個(gè)例子,你有一個(gè)為每個(gè)當(dāng)前用戶保存在session中的購物車bean,里面是表示當(dāng)前用戶決定購買物品的屬性。這個(gè)bean有一個(gè)checkOut()方法用來驗(yàn)證用戶的信用卡,將定單發(fā)給庫房以選擇貨品和出貨。別的系統(tǒng)分別地表示同樣的行為,或許使用Session EJBs。
在一些小型應(yīng)用程序中,同樣的行為又可能嵌入到作為Controller一部分的 Action 類中。這在邏輯非常簡單或者并不想要在其它環(huán)境中重用這些商業(yè)邏輯時(shí)是恰當(dāng)?shù)?。Struts框架支持所有這些方法,但建議將商業(yè)邏輯(“做什么”)和 Action 類(“決定做什么”)分離開。 

1.4 View: JSP頁面和表示組件

基于Struts的應(yīng)用程序中的 View 部分通常使用JSP技術(shù)來構(gòu)建。JSP頁面包含稱為“模版文本”的靜態(tài)HTML(或XML)文本,加上插入的基于對特殊行為標(biāo)記解釋的動(dòng)態(tài)內(nèi)容。JSP環(huán)境包括了其用途由JSP規(guī)范來描述的一套標(biāo)準(zhǔn)的行為標(biāo)記,例如  。另外,還有一個(gè)用來定義你自己標(biāo)記的標(biāo)準(zhǔn)機(jī)制,這些自定義的標(biāo)記組織在“定制標(biāo)記庫”中?!?
Struts包括了一個(gè)廣闊的便于創(chuàng)建用戶界面,并且充分國際化的定制標(biāo)記庫,與作為系統(tǒng) Model 部分一部分的ActionForm beans美妙地相互配合。這些標(biāo)記的使用將在后面做詳細(xì)討論。
除了JSP頁面和其包含的行為及定制標(biāo)記,商業(yè)對象經(jīng)常需要能夠基于它們在被請求時(shí)的當(dāng)前狀態(tài)將自己處理成HTML(或XML)。從這些對象處理過的輸出可以很容易地使用  標(biāo)準(zhǔn)行為標(biāo)記包括在結(jié)果的JSP頁面中?!?

1.5 Controller: ActionServlet和ActionMapping

應(yīng)用程序的 Controller 部分集中于從客戶端接收請求(典型情況下是一個(gè)運(yùn)行瀏覽器的用戶),決定執(zhí)行什么商業(yè)邏輯功能,然后將產(chǎn)生下一步用戶界面的責(zé)任委派給一個(gè)適當(dāng)?shù)腣iew組件。在Struts中,controller的基本組件是一個(gè) ActionServlet 類的servlet。這個(gè)servlet通過定義一組映射(由Java接口 ActionMapping 描述)來配置。每個(gè)映射定義一個(gè)與所請求的URI相匹配的路徑和一個(gè) Action 類(一個(gè)實(shí)現(xiàn) Action 接口的類)完整的類名,這個(gè)類負(fù)責(zé)執(zhí)行預(yù)期的商業(yè)邏輯,然后將控制分派給適當(dāng)?shù)腣iew組件來創(chuàng)建響應(yīng)?!?
Struts也支持使用包含有運(yùn)行框架所必需的標(biāo)準(zhǔn)屬性之外的附加屬性的 ActionMapping 類的能力。這允許你保存特定于你的應(yīng)用程序的附加信息,同時(shí)仍可利用框架其余的特性。另外,Struts允許你定義控制將重定向到的邏輯名,這樣一個(gè)行為方法可以請求“主菜單”頁面(舉例),而不需要知道相應(yīng)的JSP頁面的實(shí)際名字是什么。這個(gè)功能極大地幫助你分離控制邏輯(下一步做什么)和顯示邏輯(相應(yīng)的頁面的名稱是什么)?!?

2. 創(chuàng)建Model組件

2.1 概述

你用到的應(yīng)用程序的需求文檔很可能集中于創(chuàng)建用戶界面。然而你應(yīng)該保證每個(gè)提交的請求所需要的處理也要被清楚的定義。通常說來,Model 組件的開發(fā)者集中于創(chuàng)建支持所有功能需求的JavaBeans類。一個(gè)特殊應(yīng)用要求的beans的精確特性依賴于具體需求變化會(huì)非常的大,但是它們通??梢苑殖上旅嬗懻摰膸追N類型。然而,首先對“范圍”概念做一個(gè)簡短的回顧是有用的,因?yàn)樗cbeans有關(guān)。 

2.2 JavaBeans和范圍

在一個(gè)基于web的應(yīng)用程序中,JavaBeans可以被保存在(并從中訪問)一些不同“屬性”的集合中。每一個(gè)集合都有集合生存期和所保存的beans可見度的不同的規(guī)則??偟恼f來,定義生存期和可見度的這些規(guī)則被叫做這些beans的 范圍 。JSP規(guī)范中使用以下術(shù)語定義可選的范圍(在圓括號中定義servlet API中的等價(jià)物): 
page - 在一個(gè)單獨(dú)的JSP頁面中可見的Beans,生存期限于當(dāng)前請求。(service()方法中的局部變量) request - 在一個(gè)單獨(dú)的JSP頁面中可見的Beans,也包括所有包含于這個(gè)頁面或從這個(gè)頁面重定向到的頁面或servlet。(Request屬性) 
session - 參與一個(gè)特定的用戶session的所有的JSP和servlet都可見的Beans,跨越一個(gè)或多個(gè)請求。(Session屬性) 
application - 一個(gè)web應(yīng)用程序的所有JSP頁面和servlet都可見的Beans。(Servlet context屬性) 
記住同一個(gè)web應(yīng)用程序的JSP頁面和servlets共享同樣一組bean集合是很重要的。例如,一個(gè)bean作為一個(gè)request屬性保存在一個(gè)servlet中,就象這樣: 
MyCart mycart = new MyCart(...);
request.setAttribute("cart", mycart);
將立即被這個(gè)servlet重定向到的一個(gè)JSP頁面使用一個(gè)標(biāo)準(zhǔn)的行為標(biāo)記看到,就象這樣: 


2.3 ActionForm Beans

Struts框架通常假定你已經(jīng)為每一個(gè)你的應(yīng)用程序中請求的輸入創(chuàng)建了一個(gè) ActionForm bean(即一個(gè)實(shí)現(xiàn)了 ActionForm 接口的類)。如果你在你的 ActionMapping 配置文件中定義了這樣的beans(見“創(chuàng)建Controller組件”),Struts的controller servlet在調(diào)用適當(dāng)?shù)摹ction 方法前將自動(dòng)為你執(zhí)行如下的服務(wù): 
用適當(dāng)?shù)年P(guān)鍵字檢查用戶的session中是否有適當(dāng)?shù)念惖腷ean的一個(gè)實(shí)例?!?
如果沒有這樣的session范圍的bean,自動(dòng)建立一個(gè)新的bean并添加到用戶的session中?!?
對每個(gè)名字對應(yīng)于bean中的一個(gè)屬性的請求參數(shù),調(diào)用相應(yīng)的set方法。這個(gè)操作類似于當(dāng)你以通配符“*”選擇所有屬性使用標(biāo)準(zhǔn)的JSP行為標(biāo)記  ?!?
更新的ActionForm bean在被調(diào)用時(shí)將被傳遞給Acton類的perform()方法,以使這些值能夠立即生效?!?
當(dāng)你在寫你的ActionForm beans時(shí),記住以下的原則: 
ActionForm 接口本身不需要特殊的實(shí)現(xiàn)方法。它是用來標(biāo)識(shí)這些特定的beans在整個(gè)體系結(jié)構(gòu)中的作用。典型情況下,一個(gè)ActionForm bean只包括屬性的get方法和set方法,沒有商業(yè)邏輯?!?
通常在一個(gè)ActionForm bean中只有很少的輸入驗(yàn)證邏輯。這樣的beans存在的主要理由是保存用戶為相關(guān)的表單所輸入的大部分近期值 -- 甚至在錯(cuò)誤被檢測到時(shí) -- 這樣同樣的頁面可以被重建,伴隨有一組出錯(cuò)信息,這樣用戶僅僅需要糾正錯(cuò)誤的字段。用戶輸入的驗(yàn)證應(yīng)該在 Action 類中執(zhí)行(如果是很簡單的話),或者在適當(dāng)?shù)纳虡I(yè)邏輯beans中執(zhí)行。 
為每個(gè)表單中出現(xiàn)的字段定義一個(gè)屬性(用相關(guān)的getXxx()和setXxx()方法)。字段名和屬性名必須按照J(rèn)avaBeans的約定相匹配。例如,一個(gè)名為 username 的輸入字段將引起 setUsername() 方法被調(diào)用?!?
你應(yīng)該注意一個(gè)“表單”在這里討論時(shí)的意義并不必須對應(yīng)于用戶界面中的一個(gè)單獨(dú)的JSP頁面。在很多應(yīng)用程序中一個(gè)“表單”(從用戶的觀點(diǎn))延伸至多個(gè)頁面也是很平常的。想想看,例如,通常在安裝新的應(yīng)用程序時(shí)使用的導(dǎo)航安裝程序的用戶界面。Struts鼓勵(lì)你定義一個(gè)包含所有字段屬性的單獨(dú)的ActionForm bean。不管字段實(shí)際上是顯示在哪個(gè)頁面上。同樣的,同一表單的不同的頁面應(yīng)該提交到相同的Action類。如果你遵照這個(gè)建議,在大多數(shù)情況下,頁面設(shè)計(jì)者可以重新組織不同頁面中的字段而不需要改變處理邏輯?!?

2.4 系統(tǒng)狀態(tài)Beans

系統(tǒng)的實(shí)際狀態(tài)通常表示為一組一個(gè)或多個(gè)的JavaBeans類,其屬性定義當(dāng)前狀態(tài)。例如,一個(gè)購物車系統(tǒng)包括一個(gè)表示購物車的bean,這個(gè)bean為每個(gè)單獨(dú)的購物者維護(hù),這個(gè)bean中包括(在其它事物之中)一組購物者當(dāng)前選擇購買的項(xiàng)目。分別地,系統(tǒng)也包括保存用戶信息(包括他們的信用卡和送貨地址)、可獲得項(xiàng)目的目錄和它們當(dāng)前庫存水平的不同的beans。 
對于小規(guī)模的系統(tǒng),或者對于不需要長時(shí)間保存的狀態(tài)信息,一組系統(tǒng)狀態(tài)beans可以包含所有系統(tǒng)曾經(jīng)經(jīng)歷的特定細(xì)節(jié)的信息?;蛘呓?jīng)常是,系統(tǒng)狀態(tài)beans表示永久保存在一些外部數(shù)據(jù)庫中的信息(例如CustomerBean對象對應(yīng)于表 CUSTOMERS 中的特定的一行),在需要時(shí)從服務(wù)器的內(nèi)存中創(chuàng)建或清除。在大規(guī)模應(yīng)用程序中,Entity EJBs也用于這種用途?!?

2.5 商業(yè)邏輯Beans

你應(yīng)該把你的應(yīng)用程序中的功能邏輯封裝成對為此目的設(shè)計(jì)的JavaBeans的方法調(diào)用。這些方法可以是用于系統(tǒng)狀態(tài)beans的相同的類的一部分,或者可以是在專門執(zhí)行商業(yè)邏輯的獨(dú)立的類中。在后一種情況下,你通常需要將系統(tǒng)狀態(tài)beans傳遞給這些方法作為參數(shù)處理?!?
為了代碼最大的可重用性,商業(yè)邏輯beans應(yīng)該被設(shè)計(jì)和實(shí)現(xiàn)為它們不知道自己被執(zhí)行于web應(yīng)用環(huán)境中。如果你發(fā)現(xiàn)在你的bean中你必須import一個(gè) javax.servlet.* 類,你就把這個(gè)商業(yè)邏輯捆綁在了web應(yīng)用環(huán)境中??紤]重新組織事物使你的 Action 類(Controller任務(wù)的一部分,在下面描述)翻譯所有從HTTP請求中請求被處理為對你的商業(yè)邏輯beans屬性set方法調(diào)用的信息,然后可以發(fā)出一個(gè)對 execute() 的調(diào)用。這樣的一個(gè)商業(yè)邏輯類可以被重用在除它們最初被構(gòu)造的web應(yīng)用程序以外的環(huán)境中?!?
依賴于你的應(yīng)用程序的復(fù)雜度和范圍,商業(yè)邏輯beans可以是與作為參數(shù)傳遞的系統(tǒng)狀態(tài)beans交互作用的普通的JavaBeans,或者使用JDBC調(diào)用訪問數(shù)據(jù)庫的普通的JavaBeans。而對于較大的應(yīng)用程序,這些beans經(jīng)常是有狀態(tài)或無狀態(tài)的EJBs?!?

2.6 題外話: 訪問關(guān)系數(shù)據(jù)庫

很多web應(yīng)用程序利用一個(gè)關(guān)系數(shù)據(jù)庫(通過一個(gè)JDBC driver訪問)來保存應(yīng)用程序相關(guān)的永久數(shù)據(jù)。其它應(yīng)用程序則使用Entity EJBs來實(shí)現(xiàn)這個(gè)目的,他們委派EJBs自己來決定怎樣維護(hù)永久狀態(tài)。如果你是使用EJBs來實(shí)現(xiàn)這個(gè)目的,遵照EJB規(guī)范中描述的客戶端設(shè)計(jì)模式?!?
對于基于直接數(shù)據(jù)庫訪問的web應(yīng)用程序,一個(gè)普通的設(shè)計(jì)問題是當(dāng)需要訪問低層數(shù)據(jù)庫時(shí)怎樣產(chǎn)生一個(gè)適當(dāng)?shù)腏DBC連接對象。解決這個(gè)問題有幾種方法 -- 以下原則描述了推薦的一種方法: 
創(chuàng)建或得到一個(gè)允許一組數(shù)據(jù)庫連接被多個(gè)用戶共享的ConnectionPool類。Struts(當(dāng)前)沒有包括這樣的一個(gè)類,但是有很多這樣的類可以得到。 
當(dāng)應(yīng)用程序初始化時(shí),在應(yīng)用程序展開(deployment)描述符中定義一個(gè)有一個(gè)“啟動(dòng)時(shí)加載”值的servlet。我們將把這個(gè)servlet叫做 啟動(dòng) servlet。在大多數(shù)情況下,這個(gè)servlet不需要處理任何的請求,所以沒有一個(gè)  會(huì)指向它?!?
在啟動(dòng)servlet的 init() 方法中,配置并初始化一個(gè)ConnectionPool類的實(shí)例,將其保存為一個(gè)servlet context屬性(從JSP的觀點(diǎn)看等同于一個(gè)application范圍的bean)。通?;趥鬟f給啟動(dòng)servlet初始化參數(shù)來配置聯(lián)接緩沖池是很方便的?!?
在啟動(dòng)servlet的 destroy() 方法中,包含了釋放聯(lián)接緩沖池所打開的聯(lián)接的邏輯。這個(gè)方法將在servlet容器結(jié)束這個(gè)應(yīng)用程序的時(shí)候被調(diào)用?!?
當(dāng) Action 類需要調(diào)用一個(gè)需要數(shù)據(jù)庫聯(lián)接的商業(yè)邏輯bean中的方法(例如“insert a new customer”)時(shí),將執(zhí)行下面的步驟: 
為這個(gè)web應(yīng)用程序從servelt context屬性中得到一個(gè)聯(lián)接緩沖池對象。 
調(diào)用聯(lián)接緩沖池對象的 open() 方法來得到一個(gè)在 Action 類調(diào)用中使用的聯(lián)接?!?
調(diào)用商業(yè)邏輯bean中合適的方法,將數(shù)據(jù)庫聯(lián)接對象作為一個(gè)參數(shù)傳遞給它?!?
調(diào)用分配的聯(lián)接中的 close() 方法,這將引起這個(gè)聯(lián)接為了以后其它請求的重用被返回到緩沖池中。 
一個(gè)通常的編程錯(cuò)誤是忘記了把數(shù)據(jù)庫聯(lián)接返回給緩沖池,這將最終導(dǎo)致用完所有的聯(lián)接。一定要確信 Action 類的邏輯總是返回聯(lián)接,甚至在商業(yè)邏輯bean拋出一個(gè)違例時(shí)。 
遵照上面推薦的設(shè)計(jì)模式意味著你能夠編寫你的商業(yè)邏輯類而不需要擔(dān)心它們怎樣得到一個(gè)JDBC聯(lián)接來使用-- 簡單地在任何需要訪問數(shù)據(jù)庫的方法中包含一個(gè)Connection參數(shù)。當(dāng)你的商業(yè)邏輯類在一個(gè)web應(yīng)用程序中被利用時(shí),分配和釋放適當(dāng)?shù)穆?lián)接是 Action 類的責(zé)任。當(dāng)你使用相同的商業(yè)邏輯類時(shí),例如,在一個(gè)批處理工作中,提供一個(gè)適當(dāng)?shù)穆?lián)接是那個(gè)應(yīng)用程序的責(zé)任(這不需要從緩沖池中獲得,因?yàn)榇蠖鄶?shù)批處理工作運(yùn)行于一個(gè)單線程環(huán)境中)?!?

3. 創(chuàng)建View組件

3.1 概述

這一章集中于創(chuàng)建應(yīng)用程序中的 View 組件的任務(wù),主要使用JSP技術(shù)建立。特別的,Struts除了提供了與輸入表單的交互外還提供了建立國際化應(yīng)用程序的支持。幾個(gè)其它的與View相關(guān)的主題也被簡單地討論?!?

3.2 國際化消息

幾年之前,應(yīng)用程序開發(fā)者能夠考慮到僅僅支持他們本國的只使用一種語言(或者有時(shí)候是兩種)和通常只有一種數(shù)量表現(xiàn)方式(例如日期、數(shù)字、貨幣值)的居民。然而,基于web技術(shù)的應(yīng)用程序的爆炸性增長,以及將這些應(yīng)用程序展開在Internet或其它被廣泛訪問的網(wǎng)絡(luò)之上,已經(jīng)在很多情況下使得國家的邊界淡化到不可見。這種情況轉(zhuǎn)變成為一種對于應(yīng)用程序支持國際化(經(jīng)常被稱做“i18n”,因?yàn)?8是字母“i”和字母“n”之間的字母個(gè)數(shù))和本地化的需求?!?
Struts建立于Java平臺(tái)之上為建立國際化和本地化的應(yīng)用程序提供幫助。需要熟悉的關(guān)鍵概念是: 
Locale - 基礎(chǔ)的支持國際化的Java類是 java.util.Locale 。每個(gè) Locale 代表一個(gè)特別的國家和語言選擇(加上一個(gè)可選的語言變量),以及一套格式假定,例如數(shù)字和日期等等。 
ResourceBundle - java.util.ResourceBundle 類提供支持多種語言消息的基本工具。查看文檔中關(guān)于ResourceBundle 類以及你的JDK版本的文檔包中關(guān)于國際化的更多內(nèi)容?!?
PropertyResourceBundle - 一個(gè) ResourceBundle 類的標(biāo)準(zhǔn)實(shí)現(xiàn)允許你使用與初始化properties文件同樣的“name=value”語法來定義資源。這對于使用為用于一個(gè)web應(yīng)用程序的消息準(zhǔn)備資源包是非常方便的,因?yàn)檫@些消息通常都是面向文本的?!?
MessageFormat - java.text.MessageFormat 類允許你使用運(yùn)行時(shí)指定的參數(shù)替換一個(gè)消息字符串中的一部分(在這種情況下,是一個(gè)從一個(gè)資源包得到的消息)。這在你創(chuàng)建一個(gè)句子的場合中是有用的,但是詞會(huì)以不同的語言按照不同的順序出現(xiàn)。消息中的占位符字符串{0}用第一個(gè)運(yùn)行時(shí)參數(shù)替換,{1}用第二個(gè)運(yùn)行時(shí)參數(shù)替換,以此類推?!?
MessageResources - Struts的類 org.apache.struts.util.MessageResources 使你能夠?qū)⒁惶踪Y源包視做一個(gè)數(shù)據(jù)庫,并且允許你為一個(gè)特定的Locale(通常是與當(dāng)前用戶相對應(yīng))請求一個(gè)特定的消息,而不是為服務(wù)器運(yùn)行在其中的缺省的Locale請求消息?!?
對于一個(gè)國際化的應(yīng)用程序,遵照J(rèn)DK文檔包中國際化文檔所描述的步驟來創(chuàng)建一個(gè)包含每種語言的消息的屬性文件。下面舉一個(gè)例子說明?!?
假設(shè)你的源代碼建立在包 com.mycompany.mypackage 中,因此它保存于一個(gè)叫做(相對于你的源目錄)com/mycompany/mypackage 的目錄中。為創(chuàng)建一個(gè)叫做 com.mycompany.mypackage.MyResources 的資源包,你應(yīng)該在目錄 com/mycompany/mypackage 中創(chuàng)建下列文件: 
MyResources.properties - 包含你的服務(wù)器的缺省語言的消息。如果你的缺省語言是英語,你可能有一個(gè)這樣的條目: 
prompt.hello=Hello
MyResources_xx.properties - 包含ISO語言編程為“xx”(查看ResourceBundle的Java文檔頁面得到一個(gè)當(dāng)前列表的聯(lián)接)的同樣的消息。對于上面的消息的法語版,你可以有這個(gè)條目: 
prompt.hello=Bonjour
你可以有你需要的任意多的語言的資源包文件。
當(dāng)你在web應(yīng)用程序展開描述符中配置controller servlet時(shí),你需要在一個(gè)初始化參數(shù)中定義的一件事是應(yīng)用程序的資源包的基礎(chǔ)名。在上述的情況中,這應(yīng)該是 com.mycompany.mypackage.MyResources ?!?

3.3 表單和FormBean的交互

大部分web開發(fā)者曾經(jīng)使用HTML的標(biāo)準(zhǔn)性能來建立表單,例如使用  標(biāo)記。用戶希望交互程序具有一定的行為,這些期待中的一個(gè)與錯(cuò)誤處理有關(guān) -- 如果用戶出現(xiàn)一個(gè)錯(cuò)誤,應(yīng)用程序應(yīng)該允許他們僅僅修改需要修改的部分 -- 而不需要重新敲入當(dāng)前頁面或表單中的任何其它信息?!?
使用標(biāo)準(zhǔn)的HTML和JSP編程來完全實(shí)現(xiàn)這個(gè)期望是單調(diào)而繁重的。舉例來說,一個(gè)用戶名字段的輸入元素看起來可以象是這樣(在JSP中) 
value="<%= loginBean.getUsername() %>">
這很難敲對,會(huì)把沒有編程概念的HTML開發(fā)者搞糊涂,并且會(huì)在HTML編輯器中造成問題。取而代之的是,Struts提供了一種全面的基于JSP 1.1的定制標(biāo)記庫功能的機(jī)制來建立表單。上面的情況使用Struts處理后將象是這樣: 

沒有必要再顯式地涉及到從中獲得初始值的JavaBean。這將由框架自動(dòng)處理?!?

3.3.1 使用Struts建立表單

一個(gè)完整的注冊表單將演示Struts相對于直接使用HTML和標(biāo)準(zhǔn)的JSP功能怎樣極大地減輕了處理表單的痛苦??紤]以下稱為logon.jsp的頁面(來自Struts的例子程序): 
<%@ page language="java" %>
<%@ taglib uri="/WEB-INF/struts.tld" prefix="struts" %>


<struts:message key="logon.title"/>




type="org.apache.struts.example.LogonForm"/>


































下面的條目基于這個(gè)例子演示在Struts中處理表單的關(guān)鍵的特性: 
taglib指令告訴JSP頁面編譯器從哪里找到Struts標(biāo)記庫的 標(biāo)記庫描述符 。在這種情況下,我們使用struts作為前綴來標(biāo)識(shí)來自這個(gè)庫中的標(biāo)記,但是可以使用任何你想用的前綴?!?
這個(gè)頁面使用了幾個(gè) message 標(biāo)記來從一個(gè)包含有這個(gè)應(yīng)用程序所有資源的 MessageResources 對象中查找國際化的消息字符串。為了讓這個(gè)頁面能夠工作,以下的消息關(guān)鍵字必須在這些資源中被定義: 
logon.title - 注冊頁面的標(biāo)題 
prompt.username - 一個(gè) “Username:” 提示字符串 
prompt.password - 一個(gè) “Password:” 提示字符串 
button.submit - “Submit”按鈕的標(biāo)簽 
button.reset - “Reset”按鈕的標(biāo)簽 
當(dāng)用戶注冊時(shí),應(yīng)用程序可以在用戶的session中保存一個(gè) Locale 對象。這個(gè) Locale 將用來選擇適當(dāng)語言的消息。這使得給用戶一個(gè)切換語言的可選項(xiàng)實(shí)現(xiàn)起來變的容易了 -- 僅僅改變保存的 Locale 對象,所有的消息就會(huì)自動(dòng)切換?!?
errors 標(biāo)記顯示由一個(gè)商業(yè)邏輯組件保存的任何出錯(cuò)消息,或者如果沒有出錯(cuò)消息保存就什么都沒有。這個(gè)標(biāo)記將在下面做深入的描述?!?
form 標(biāo)記基于指定的屬性對一個(gè)HTML 
 元素進(jìn)行處理。它也將所有在這個(gè)表單中的字段與一個(gè)保存在關(guān)鍵字 logonForm 下的session范圍的FormBean相關(guān)聯(lián)。這個(gè)bean用來為所有的具有與bean中的屬性名匹配的名字的輸入字段提供初始值。如果適當(dāng)?shù)腷ean沒有找到,一個(gè)新的bean將會(huì)被自動(dòng)建立,使用指定的Java類名?!?
text 標(biāo)記對一個(gè)類型為“text”的HTML  元素進(jìn)行處理。在這種情況下,占據(jù)瀏覽器屏幕的字符位置的數(shù)字也被指定。當(dāng)頁面被執(zhí)行時(shí),是相對應(yīng)的bean的 username 屬性的當(dāng)前值(也就是 getUsername() 的返回值)?!?
password 標(biāo)記使用方法類似。不同之處是當(dāng)用戶敲入他們的口令時(shí)瀏覽器將回應(yīng)星號字符,而不是輸入值?!?
submit 和 reset 標(biāo)記在表單低部生成相應(yīng)的按鈕。每個(gè)按鈕的文本標(biāo)簽使用 message 標(biāo)記建立,同時(shí)帶有提示,這樣這些值就是國際化的?!?

3.3.2 輸入字段類型支持

Struts為所有以下類型的輸入字段定義了標(biāo)記,帶有與其相應(yīng)的參考信息的超聯(lián)接?!?
checkboxes 
hidden 字段 
password 輸入字段 
radio 按鈕 
reset 按鈕 
select 列表和嵌入的 
options 
submit 按鈕 
text 輸入字段 
textareas 
在所有情況下,一個(gè)字段標(biāo)記都必須嵌套在一個(gè) form 標(biāo)記中,這樣字段才知道使用哪個(gè)bean來初始化顯示的值?!?

3.3.3 其它有用的表示標(biāo)記

在Struts的標(biāo)記庫中有幾個(gè)其它的標(biāo)記對于建立用戶界面是有幫助的: 
enumerate 為一個(gè)指定集合的每個(gè)元素重復(fù)一次標(biāo)記體(可以是一個(gè)Enumeration,一個(gè)Hashtable,一個(gè)Vector或一個(gè)對象數(shù)組)?!?
getProperty 從指定的bean中得到指定的屬性,并且在本頁面的其余部分作為一個(gè)page范圍的bean存在。這是訪問一個(gè)被 enumerate 使用的集合的方便的方法?!?
ifAttributeExists 只有在一個(gè)指定的屬性存在于一個(gè)指定的范圍中時(shí)才對標(biāo)記體求值?!?
ifAttributeMissing 只有在一個(gè)指定的屬性不存在于一個(gè)指定的范圍中時(shí)才對標(biāo)記體求值?!?
ifParameterEquals 只有在一個(gè)指定的請求參數(shù)具有一個(gè)指定的值時(shí)才對標(biāo)記體求值。 
ifParameterNotEquals 只有在一個(gè)指定的請求參數(shù)不具有一個(gè)指定的值或者不存在時(shí)才對標(biāo)記體求值?!?
ifParameterNotNull 只有在一個(gè)指定的請求參數(shù)包含在這個(gè)請求中并且長度大于0時(shí)才對標(biāo)記體求值?!?
ifParameterNull 只有在一個(gè)指定的請求參數(shù)不包含在這個(gè)請求中或者長度等于0時(shí)才對標(biāo)記體求值?!?
iterate 為一個(gè)指定集合中的每個(gè)元素重復(fù)一次標(biāo)記體(可以是一個(gè)Collection,一個(gè)Iterator,一個(gè)Map,或者一個(gè)對象數(shù)組)。這個(gè)標(biāo)記在Java2環(huán)境中代替了 enumerate 標(biāo)記?!?
link 生成一個(gè)超聯(lián)接,當(dāng)沒有cookie支持時(shí)自動(dòng)應(yīng)用URL編程來維護(hù)session狀態(tài)?!?
parameter 處理指定請求參數(shù)的值,適當(dāng)?shù)剡^濾HTML中有特殊含義的字符?!?
property 顯示一個(gè)表單中命名的bean屬性 -- 在屬性應(yīng)該是只讀時(shí)使用這個(gè)標(biāo)記而不是 text 標(biāo)記。 

3.3.4 自動(dòng)表單驗(yàn)證

除了上面描述的表單和bean的交互外,如果你的bean知道怎樣驗(yàn)證它接收的輸入字段,Struts還提供一種附加的機(jī)制。為了利用這個(gè)特性,使你的bean類實(shí)現(xiàn) ValidatingActionForm 接口,而不是 ActionForm 接口。一個(gè) ValidatingActionForm 增加了一個(gè)附加的方法簽名: 
public String[] validate()
對于一個(gè)被controller servlet在bean屬性已經(jīng)組裝但是在相應(yīng)的行為類的 perform() 方法被調(diào)用之前調(diào)用的方法,validate() 方法有以下可選項(xiàng): 
執(zhí)行適當(dāng)?shù)尿?yàn)證發(fā)現(xiàn)沒有錯(cuò)誤 -- 返回 null 或者一個(gè)非0長度字符串?dāng)?shù)組,并且controller servlet將繼續(xù)調(diào)用適當(dāng)?shù)摹ction 類的 perform() 方法?!?
執(zhí)行適當(dāng)?shù)尿?yàn)證發(fā)現(xiàn)有錯(cuò)誤 -- 返回一個(gè)內(nèi)容為應(yīng)該被顯示的出錯(cuò)消息關(guān)鍵字(進(jìn)入應(yīng)用程序的MessageResources 包)的字符串?dāng)?shù)組。controller servlet將作為適合于  標(biāo)記使用的請求屬性保存這個(gè)數(shù)組,并且將控制重定向回輸入表單(由這個(gè) ActionMapping 的 inputForm 屬性標(biāo)識(shí))?!?
正如以前提到的,這個(gè)特性完全是可選的。如果你的form bean 僅僅實(shí)現(xiàn)了 ActionForm 接口,controller servlet將假設(shè)任何請求的驗(yàn)證由action類完成?!?

3.4 其它的表示技術(shù)

盡管你的應(yīng)用程序的外表和感覺可以完全基于標(biāo)準(zhǔn)的JSP能力和Struts的定制標(biāo)記庫構(gòu)建,你也應(yīng)該考慮展開其它改進(jìn)組件重用、減少管理負(fù)擔(dān)或者減少出錯(cuò)的技術(shù)。在下面的部分討論幾個(gè)可選的技術(shù)?!?

3.4.1 特定于應(yīng)用程序的定制標(biāo)記

在使用Struts庫提供的定制標(biāo)記之外,很容易建立特定于你創(chuàng)建的應(yīng)用程序的標(biāo)記來幫助建立用戶界面。Struts包括的例子程序用建立以下僅用于實(shí)現(xiàn)這個(gè)應(yīng)用程序的標(biāo)記演示了這個(gè)原則: 
checkLogon - 檢查一個(gè)特殊的會(huì)話對象的存在,如果不存在將控制重定向到注冊頁面。這是用來捕捉這樣的情況,用戶在你的應(yīng)用程序執(zhí)行的中間把一個(gè)頁面做成書簽并且試圖跳過注冊,或者用戶的會(huì)話超時(shí)?!?
linkSubscription - 為一個(gè)詳細(xì)的定單頁面生成一個(gè)超聯(lián)接,它將需要的主關(guān)鍵字值作為一個(gè)請求屬性傳遞。這在列出與一個(gè)用戶相關(guān)的定單并提供編輯或刪除定單的聯(lián)接時(shí)使用?!?
linkUser - 為一個(gè)用戶的一個(gè)具體的頁面生成一個(gè)超聯(lián)接,它將它將需要的主關(guān)鍵字值作為一個(gè)請求屬性傳遞?!?
這些標(biāo)記的源代碼在 src/example 目錄中,在包 org.apache.struts.example 里,還帶有一些其它的用在這個(gè)應(yīng)用程序中的Java類?!?

3.4.2 有包含文件的頁面組件

在一個(gè)JSP文件(包含定制標(biāo)記和beans用來訪問請求的動(dòng)態(tài)數(shù)據(jù))中創(chuàng)建完整的表示是一種非常普通的設(shè)計(jì)方法,在Struts包括的例子程序中被采用。然而很多應(yīng)用程序要求在單獨(dú)一個(gè)頁面中顯示你的應(yīng)用程序的多個(gè)邏輯上獨(dú)立的部分?!?
舉例來說,一個(gè)入口應(yīng)用程序可以在入口的主頁面上有一些或者全部以下的功能: 
訪問這個(gè)入口的一個(gè)搜索引擎?!?
一個(gè)或更多的“提供新聞”的顯示,含有按照用戶的注冊信息定制的感興趣的標(biāo)題。 
訪問與這個(gè)入口相關(guān)的討論的主題。 
如果你的入口提供免費(fèi)郵件帳號,還要有一個(gè)“郵件等待”的提示。 
如果你能夠?qū)⒐ぷ鲃澐珠_,分配不同的開發(fā)者去做不同的片段,那么這個(gè)站點(diǎn)不同片段的開發(fā)就會(huì)更加簡單。然后,你可以使用JSP技術(shù)的 include 能力來將這些片段組合進(jìn)一個(gè)單獨(dú)的頁面。有兩種 include 可用,依賴于你希望輸出的組合發(fā)生在什么時(shí)間: 
include 指令?。?%@ include file="xxxxx" %>)在JSP頁面被編譯時(shí)處理。它用于包括不需要在請求時(shí)改變的HTML代碼。它把包括進(jìn)來的文本當(dāng)作靜態(tài)文本,很象C或C++中的 #include 指令?!?
include 行為?。?jsp:include page="xxxxx" flush="true" />)在請求時(shí)處理,并且是由服務(wù)器透明處理。這意味著你可以通過把它嵌套在一個(gè)類似ifParameterEquals的標(biāo)記中有條件地執(zhí)行include ?!?

3.4.3 圖片處理組件

一些應(yīng)用程序要求動(dòng)態(tài)生成圖片,就象一個(gè)股市報(bào)告站點(diǎn)的價(jià)格圖一樣。通常使用兩種不同的方法來實(shí)現(xiàn)這個(gè)需求: 
處理一個(gè)執(zhí)行一個(gè)servlet請求的URL的超聯(lián)接。這個(gè)servlet將使用一個(gè)圖象庫來生成圖片,設(shè)置適當(dāng)?shù)腸ontent類型(例如 image/gif),并且將圖片的字節(jié)流發(fā)送回瀏覽器。瀏覽器就會(huì)象從一個(gè)靜態(tài)文件中接收到的一樣顯示圖片?!?
處理HTML代碼需要下載一個(gè)創(chuàng)建請求的圖象的Java applet。你可以通過為在處理的代碼中的這個(gè)applet設(shè)置適當(dāng)?shù)某跏蓟瘏?shù)配置這個(gè)圖象,或者你可以讓這個(gè)applet與服務(wù)器建立自己聯(lián)接來接收這些參數(shù)。 

4. 創(chuàng)建Controller組件

4.1 概述

現(xiàn)在我們理解了怎樣構(gòu)造你的應(yīng)用程序的Model和View組件,現(xiàn)在是集中到 Controller 組件的時(shí)候了。Struts包括一個(gè)實(shí)現(xiàn)映射一個(gè)請求URI到一個(gè)行為類的主要功能的servlet。因此你的與Controller有關(guān)的主要責(zé)任是: 
為每一個(gè)可能接收的邏輯請求寫一個(gè) Action 類(也就是,一個(gè) Action 接口的實(shí)現(xiàn)) 
寫一個(gè)定義類名和與每個(gè)可能的映射相關(guān)的其它信息的 ActionMapping 類(也就是,一個(gè) ActionMapping 接口的實(shí)現(xiàn)) 
寫行為映射配置文件(用XML)用來配置controller servlet?!?
為你的應(yīng)用程序更新web應(yīng)用程序展開描述符文件(用XML)用來包括必需的Struts組件?!?
給你的應(yīng)用程序添加適當(dāng)?shù)腟truts組件?!?

4.2 Action類

Action 接口定義一個(gè)單一的必須由一個(gè) Action 類實(shí)現(xiàn)的方法,就象下面這樣: 
public ActionForward perform(ActionServlet servlet,
ActionMapping mapping,
ActionForm form,
HttpServletRequest request,
HttpServletResponse response)
throws IOException, ServletException;
一個(gè) Action 類的目標(biāo)是處理這個(gè)請求,然后返回一個(gè)標(biāo)識(shí)JSP頁面的 ActionForward 對象,控制應(yīng)該重定向這個(gè)JSP頁面以生成相應(yīng)的響應(yīng)。在 Model 2 設(shè)計(jì)模式中,一個(gè)典型的 Action 類將在它的 perform() 方法中實(shí)現(xiàn)下面的邏輯: 
驗(yàn)證用戶session的當(dāng)前狀態(tài)(例如,檢查用戶已經(jīng)成功地注冊)。如果 Action 類發(fā)現(xiàn)沒有注冊存在,請求應(yīng)該重定向到顯示用戶名和口令用于注冊的JSP頁面。應(yīng)該這樣做是因?yàn)橛脩艨赡茉噲D從“中間”(也就是,從一個(gè)書簽)進(jìn)入你的應(yīng)用程序,或者因?yàn)閟ession已經(jīng)超時(shí)并且servlet容器創(chuàng)建了一個(gè)新的session。 
如果驗(yàn)證還沒有發(fā)生(由于使用一個(gè)實(shí)現(xiàn) ValidatingActionForm 接口的form bean),驗(yàn)證這個(gè) form bean 的屬性是必須的。如果發(fā)現(xiàn)一個(gè)問題,當(dāng)作一個(gè)請求屬性保存合適的出錯(cuò)信息關(guān)鍵字,然后將控制重定向回輸入表單這樣錯(cuò)誤可以被糾正?!?
執(zhí)行要求的處理來處理這個(gè)請求(例如在數(shù)據(jù)庫里保存一行)。這可以用嵌入 Action 類本身的代碼來完成,但是通常應(yīng)該調(diào)用一個(gè)商業(yè)邏輯bean的一個(gè)合適的方法來執(zhí)行?!?
更新將用來創(chuàng)建下一個(gè)用戶界面頁面的服務(wù)器端對象(典型情況下是request范圍或session范圍beans,定義你需要在多長時(shí)間內(nèi)保持這些項(xiàng)目可獲得)?!?
返回一個(gè)標(biāo)識(shí)生成響應(yīng)的JSP頁面的適當(dāng)?shù)摹ctionForward 對象,基于新近更新的beans。典型情況下,你將通過在你接收到的 ActionMapping 對象(如果你使用一個(gè)局部于與這個(gè)映射上的邏輯名)或者在controller servlet 本身(如果你使用一個(gè)全局于應(yīng)用程序的邏輯名)上調(diào)用 findForward() 得到一個(gè)對這樣一個(gè)對象的引用?!?
當(dāng)為 Action 類編程時(shí)要記住的設(shè)計(jì)要點(diǎn)包括以下這些: 
controller servlet僅僅創(chuàng)建一個(gè)你的 Action 類的實(shí)例,用于所有的請求。這樣你需要編寫你的 Action 類使其能夠在一個(gè)多線程環(huán)境中正確運(yùn)行,就象你必須安全地編寫一個(gè)servlet的 service() 方法一樣?!?
幫助線程安全編程的最重要的原則就是在你的 Action 類中僅僅使用局部變量而不是實(shí)例變量。局部變量創(chuàng)建于一個(gè)分配給(由你的JVM)每個(gè)請求線程的棧中,所以沒有必要擔(dān)心會(huì)共享它們?!?
盡管不應(yīng)該,代表你的系統(tǒng)中Model部分的的beans仍有可能拋出違例。你應(yīng)該在你的 perform() 方法的邏輯中捕捉所有這樣的違例,并且通過執(zhí)行以下語句將它們記錄在應(yīng)用程序的日志文件中(包括相應(yīng)的棧跟蹤信息): 
servlet.log("Error message text", exception);
作為一個(gè)通用的規(guī)則,分配很少的資源并在來自同一個(gè)用戶(在用戶的session中)的請求間保持它們會(huì)導(dǎo)致可伸縮性的問題。你應(yīng)該在將控制重定向到適當(dāng)?shù)腣iew組件前努力釋放這樣的資源(例如數(shù)據(jù)庫聯(lián)接) -- 甚至在你調(diào)用的一個(gè)bean拋出了一個(gè)違例時(shí)?!?
另外,你將會(huì)想要防止出現(xiàn)非常大的 Action 類。最簡單的實(shí)現(xiàn)途徑是將你的功能邏輯嵌入到 Action 類本身,而不是將其寫在獨(dú)立的商業(yè)邏輯beans中。除了使 Action 類難于理解和維護(hù)外,這種方法也使得難于重用這些商業(yè)邏輯代碼,因?yàn)榇a被嵌入到一個(gè)組件(Action 類)中并被捆綁運(yùn)行于web應(yīng)用程序環(huán)境中。 
包括在Struts中的例子程序某種程度上延伸了這個(gè)設(shè)計(jì)原則,因?yàn)樯虡I(yè)邏輯本身是嵌入到 Action 類中的。這應(yīng)該被看作是在這個(gè)樣本應(yīng)用程序設(shè)計(jì)中的一個(gè)bug,而不是一個(gè)Struts體系結(jié)構(gòu)中的固有特性,或者是一個(gè)值得仿效的方法。 

4.3 ActionMapping實(shí)現(xiàn)

為了成功地運(yùn)行,Struts的controller servlet需要知道關(guān)于每個(gè)URI該怎樣映射到一個(gè)適當(dāng)?shù)摹ction 類的幾件事。需要了解的知識(shí)封裝在一個(gè)叫做 ActionMapping 的Java接口中,它有以下屬性: 
actionClass - 用于這個(gè)映射的 Action 類完整的Java類名。第一次一個(gè)特定的映射被使用,一個(gè)這個(gè)類的實(shí)例將被創(chuàng)建并為以后重用而保存?!?
formAttribute - session范圍的bean的名字,當(dāng)前的這個(gè)映射的 ActionForm 被保存在這個(gè)bean之下。如果這個(gè)屬性沒有被定義,沒有 ActionForm 被使用?!?
formClass - 用于這個(gè)映射的 ActionForm 類完整的Java類名。如果你在使用對form beans的支持,這個(gè)類的一個(gè)實(shí)例將被創(chuàng)建并保存(在當(dāng)前的用戶會(huì)話中) 
path - 匹配選擇這個(gè)映射的請求的URI路徑??聪旅嫒绾纹ヅ涞睦??!?
Struts在一個(gè)叫做 ActionMappingBase 的類中包括了一個(gè) ActionMapping 接口的方便的實(shí)現(xiàn)。如果你不需要為你自己的映射定義任何附加的屬性,盡管把這個(gè)類作為你的 ActionMapping 類好了,就向下面部分描述的那樣配置。然而,定義一個(gè) ActionMapping 實(shí)現(xiàn)(多半是擴(kuò)展 ActionMappingBase 類)來包含附加的屬性也是可能的。controller servlet知道怎樣自動(dòng)配置這些定制屬性,因?yàn)樗褂肧truts的Digester模塊來讀配置文件?!?
包括在Struts的例子程序中,這個(gè)特性用來定義兩個(gè)附加的屬性: 
failure - 如果Action類檢測到它接收的輸入字段的一些問題,控制應(yīng)該被重定向到的上下文相關(guān)的URI。典型情況下是請求發(fā)向的JSP頁面名,它將引起表單被重新顯示(包含Action類設(shè)置的出錯(cuò)消息和大部分最近的來自ActionForm bean的輸入值)?!?
success - 如果Action類成功執(zhí)行請求的功能,控制應(yīng)該被重定向到的上下文相關(guān)的URI。典型情況下是準(zhǔn)備這個(gè)應(yīng)用程序的會(huì)話流的下一個(gè)頁面的JSP頁面名。 
使用這兩個(gè)額外的屬性,例子程序中的 Action 類幾乎完全獨(dú)立于頁面設(shè)計(jì)者使用的實(shí)際的JSP頁面名。這個(gè)頁面可以在重新設(shè)計(jì)時(shí)被重命名,然而幾乎不會(huì)影響到 Action 類本身。如果“下一個(gè)”JSP頁面的名字被硬編碼到 Action 類中,所有的這些類也需要被修改?!?

4.4 Action映射配置文件

controller servlet怎樣知道你想要得到的映射?寫一個(gè)簡單地初始化新的 ActionMapping 實(shí)例并且調(diào)用所有適當(dāng)?shù)膕et方法的小的Java類是可能的(但是很麻煩)。為了使這個(gè)處理簡單些,Struts包括一個(gè)Digester模塊能夠處理一個(gè)想得到的映射的基于XML的描述,同時(shí)創(chuàng)建適當(dāng)?shù)膶ο?。看 API 文檔 以獲得關(guān)于Digester更多的信息?!?
開發(fā)者的責(zé)任是創(chuàng)建一個(gè)叫做 action.xml 的XML文件,并且把它放在你的應(yīng)用程序的WEB-INF目錄中。(注意這個(gè)文件并不需要 DTD,因?yàn)閷?shí)際使用的屬性對于不同的用戶可以是不同的)最外面的XML元素必須是,在這個(gè)元素之中是嵌入的0個(gè)或更多的  元素 -- 每一個(gè)對應(yīng)于你希望定義的一個(gè)映射?!?

來自例子程序的 action.xml 文件包括“注冊”功能的以下映射條目,我們用來說明這個(gè)需求: 





actionClass="org.apache.struts.example.LogonAction"
formAttribute="logonForm"
formClass="org.apache.struts.example.LogonForm"
inputForm="/logon.jsp">




就象你所看到的,這個(gè)映射匹配路徑 /logon?。▽?shí)際上,因?yàn)槔映绦蚴褂脭U(kuò)展匹配,你在一個(gè)JSP頁面指定的請求的URI結(jié)束于/logon.do)。當(dāng)接收到一個(gè)匹配這個(gè)路徑的請求時(shí),一個(gè) LogonAction 類的實(shí)例將被創(chuàng)建(僅僅在第一次)并被使用。controller servlet將在關(guān)鍵字 logonForm 下查找一個(gè)session范圍的bean,如果需要就為指定的類創(chuàng)建并保存一個(gè)bean。 
這個(gè) action 元素也定義了一個(gè)邏輯名“success”,它在 LogonAction 類中被用來標(biāo)識(shí)當(dāng)一個(gè)用戶成功注冊時(shí)使用的頁面。象這樣使用一個(gè)邏輯名允許將 action 類隔離于任何由于重新設(shè)計(jì)位置而可能發(fā)生的頁面名改變?!?

這是第二個(gè)在任何 action 之外宣告的 forward 元素,這樣它就可以被所有的action全局地獲得。在這個(gè)情況下,它為注冊頁面定義了一個(gè)邏輯名。當(dāng)你調(diào)用 mapping.findForward() 時(shí)在你的 action 代碼中,Struts首先查找這個(gè)action本地定義的邏輯名。如果沒有找到,Struts會(huì)自動(dòng)為你查找全局定義的邏輯名?!?

4.5 Web應(yīng)用程序展開描述符

設(shè)置應(yīng)用程序最后的步驟是配置應(yīng)用程序展開描述符(保存在文件WEB-INF/web.xml中)以包括所有必需的Struts組件。作為一個(gè)指南使用例子程序的展開描述符,我們看到下面的條目需要被創(chuàng)建或修改?!?

4.5.1 配置Action Servlet實(shí)例

添加一個(gè)條目定義action servlet本身,同時(shí)包括適當(dāng)?shù)某跏蓟瘏?shù)。這樣一個(gè)條目看起來象是這樣: 

action
org.apache.struts.action.ActionServlet
aram>
application
org.apache.struts.example.ApplicationResources


config
/WEB-INF/action.xml


debug
2


mapping
org.apache.struts.example.ApplicationMapping

2

controller servlet支持的初始化參數(shù)在下面描述,拷貝自 ActionServlet 類的 Javadocs 。方括號描述如果你沒有為那個(gè)初始化參數(shù)提供一個(gè)值時(shí)假設(shè)的缺省值?!?
application - 應(yīng)用程序資源包基類的Java類名。[NONE]. 
config - 包含配置信息的XML資源的上下文相關(guān)的路徑。[/WEB-INF/action.xml] 
debug - 這個(gè)servlet的調(diào)試級別,它控制記錄多少信息到日志中。[0] 
digester - 我們在 initMapping() 中利用的Digester的調(diào)試級別,它記錄到System.out而不是servlet的日志中。[0] 
forward - 使用的ActionForward實(shí)現(xiàn)的Java類名。[org.apache.struts.action.ActionForward] 
mapping - 使用的ActionMapping實(shí)現(xiàn)的Java類名。[org.apache.struts.action.ActionMappingBase] 
nocache - 如果設(shè)置為 true,增加HTTP頭信息到所有響應(yīng)中使瀏覽器對于生成或重定向到的任何響應(yīng)不做緩沖。[false] 
null - 如果設(shè)置為 true,設(shè)置應(yīng)用程序資源使得如果未知的消息關(guān)鍵字被使用則返回 null。否則,一個(gè)包括不歡迎的消息關(guān)鍵字的出錯(cuò)消息將被返回。[true] 

4.5.2 配置Action Servlet映射

有兩種通常的方法來定義將被controller servlet處理的URL -- 前綴匹配和擴(kuò)展匹配。每種方法的一個(gè)適當(dāng)?shù)挠成錀l目將在下面被描述。 
前綴匹配意思是你想讓所有以一個(gè)特殊值開頭(在上下文路徑部分之后)的URL傳遞給這個(gè)servlet。這樣一個(gè)條目看起來可以象是這樣: 


action
/execute/*

它意味著一個(gè)匹配前面描述的 /logon 路徑的請求的URL看起來象是這樣: 
http://www.mycompany.com/myapplication/execute/logon
這里 /myapplicationis 是你的應(yīng)用程序展開所在的上下文路徑。 
另一方面,擴(kuò)展映射基于URL以一個(gè)跟著定義的一組字符的句點(diǎn)結(jié)束的事實(shí)而將URL匹配到action servlet 。例如,JSP處理servlet映射到 *.jsp 模式這樣它在每個(gè)JSP頁面請求時(shí)被調(diào)用。為了使用 *.do 擴(kuò)展(它意味著“做某件事”)映射條目看起來應(yīng)該象是這樣: 


action
*.do

并且一個(gè)匹配以前描述的 /logon 路徑的請求的URI可以看起來象是這樣: 
http://www.mycompany.com/myapplication/logon.do

4.5.3 配置Struts標(biāo)記庫

下一步,你必須添加一個(gè)定義Struts標(biāo)記庫的條目。這個(gè)條目看起來應(yīng)該象是這樣: 

/WEB-INF/struts.tld
/WEB-INF/struts.tld

它告訴JSP系統(tǒng)到哪里去找這個(gè)庫的標(biāo)記庫描述符(在你的應(yīng)用程序的WEB-INF目錄,而不是在外部互聯(lián)網(wǎng)上的某個(gè)地方)。 

4.5.4 添加Struts組件到你的應(yīng)用程序中

為了在你的應(yīng)用程序運(yùn)行時(shí)使用Struts,你必須將 struts.tld 文件拷貝到你的 WEB-INF 目錄,將 struts.jar 文件拷貝到你的 WEB-INF/lib ?!? 



(2)
了解MVC

MVC是Model,View,Controller的縮寫,MVC是Application開發(fā)的設(shè)計(jì)模式,也就是大家
所知道的Model2.在MVC的設(shè)計(jì)模式中,要求在Application開發(fā)中你把商業(yè)邏輯,界面
顯示,數(shù)據(jù)分離。也就是分別在Model,View,Controller實(shí)現(xiàn):數(shù)據(jù),控制(商業(yè)邏輯)
顯示(頁面顯示).
在以前或者說傳統(tǒng)的Web Application開發(fā)方式當(dāng)中,如Asp,Php,Jsp(Model 1)開發(fā)當(dāng)中,我們在Asp(Php,Jsp)中實(shí)現(xiàn)一切,如:從數(shù)據(jù)庫中取到我們需要的數(shù)據(jù),并根據(jù)數(shù)據(jù)之間的關(guān)聯(lián)和實(shí)際的需要按照某種方式把他顯示在頁面中以及從頁面提交的表單中提取數(shù)據(jù),根據(jù) 商業(yè)邏輯從數(shù)據(jù)庫查詢相關(guān)數(shù)據(jù),或者把數(shù)據(jù)寫入數(shù)據(jù)庫。也就是說我們在Asp(Php,Jsp) 實(shí)現(xiàn)一切包括:界面顯示,商業(yè)邏輯,數(shù)據(jù)存取。這樣帶來的后果就是你所寫的Asp(Php,Jsp) 沒有層次,并且Html和Script(JavaScript、JScript,Asp、Php、Jsp源代碼)相互嵌套.可 維護(hù)性差,最要命的是在Web Application通常顯示一塊是由美工完成的,很多時(shí)候也是 你先寫好Asp、Php、Jsp然后美工進(jìn)行美化,很有可能你發(fā)現(xiàn)經(jīng)過美工處理完以后你的代碼 已經(jīng)面目全非了。你不得不把你的代碼重新組織。
在MVC模式中這個(gè)問題的解決辦法是:View中負(fù)責(zé)顯示,View一般從Controller得到已經(jīng)處理 過的數(shù)據(jù),然后顯示在頁面當(dāng)中,應(yīng)該說這樣在Html中嵌套很少的Script.基本上美工的修改 不大會(huì)廢掉你的勞動(dòng)成果。
在使用Java開發(fā)Web Application有幾種符合MVC設(shè)計(jì)模式的開發(fā)方式讓你選擇。
1:Jsp+Servlet+JavaBean(EJB)
2:Jsp+JavaBean(Controller)+JavaBean(EJB)(Model)
3:TDK(Turbine,Velocity...)
4:Xsp
5:Jsp+Struts+JavaBean(EJB)
我個(gè)人認(rèn)為后面兩種比較好,其他幾種都有可取的地方特別是使用TDK因?yàn)橛幸粋€(gè)比較好的
工具可以自動(dòng)生成很多代碼,至于它的缺點(diǎn)在后面幾種開發(fā)方式的比較當(dāng)中我會(huì)介紹。

Struts1.1的新功能
Struts1.1與1.0相比加了一些很不錯(cuò)的功能。最主要是表單驗(yàn)證上功能增強(qiáng)。在Struts1.1
數(shù)據(jù)的驗(yàn)證不象以前在Action中在validator具體實(shí)現(xiàn),而是在validation.xml通過配置實(shí)現(xiàn)
這樣做的好處就是重用性加強(qiáng)了很多。

Struts1.1實(shí)現(xiàn)的主要組成
主要包括:Action,ActionForm,ActionMapping,ActionForward,開發(fā)當(dāng)中最主要寫的是Action
ActionForm根據(jù)需要可以寫或不寫。下面我就一一具體介紹。
Action
An Action is an adapter between the contents of an incoming HTTP request
and the corresponding business logic that should be executed to process this
request.
上面是Struts開發(fā)小組對Action的描述,說Action實(shí)際上是Request和Business Logic
中間的適配器.通俗的說就是從表單中取到數(shù)據(jù)并穿給商業(yè)邏輯操作進(jìn)行一系列的操作
然后返回相應(yīng)的操作信息。

ActionForm
An ActionForm is a JavaBean optionally associated with one or more
ActionMappings. Such a bean will have had its properties initialized from
the corresponding request parameters before the corresonding action's execute()
method is called.
ActionForm實(shí)際上就是把從Request取到的數(shù)據(jù)封裝并進(jìn)行校驗(yàn),然后把合法的數(shù)據(jù)給
Action進(jìn)行處理。實(shí)際上ActionForm除了進(jìn)行數(shù)據(jù)校驗(yàn)之外另外更重要的是在表單回寫
的時(shí)候作用很大。反而在1.1以后數(shù)據(jù)校驗(yàn)的大部分工作在validation.xml去實(shí)現(xiàn)。

ActionMapping,ActionForward
ActionMapping主要是用與配置和描述相關(guān)屬性使用的。先看下在struts-config.xml
中的配置文件一段配置描述:


type="com.bingo.finance.action.UseregAction"
name="useregForm"
scope="request"
validate="true"
input="/usereg.jsp">



ActionMapping就是用來描述一個(gè)Action的URL、具體實(shí)現(xiàn)的文件、相對應(yīng)的ActionForm
數(shù)據(jù)屬性(request or session)、是否需要進(jìn)行數(shù)據(jù)校驗(yàn)和回寫、以及處理完成后可能
跳轉(zhuǎn)的URL.
而ActionForward你就可以理解為Action 操作完成后的跳轉(zhuǎn)URL,Action在處理完相關(guān)操作后
返回的是一個(gè)ActionForward也就是告訴Struts我做完這個(gè)操作下一步到哪兒去。


構(gòu)建Struts1.1運(yùn)行環(huán)境
我的配置是居于Tomcat4.0以上版本討論,其他的AppServer大致相同。
1:得到Struts1.1
http://jakarta.apache.org/builds/jakarta-struts/release/v1.1-b1/jakarta-struts-1.1-b1.zip
2:設(shè)置
把Struts.jar Copy到$Tomcat_home/common/lib 或你使用Struts的Appaction下的WEB-INF/lib下
在你使用Struts的Appaction下web.xml中增加下列配置


action
org.apache.struts.action.ActionServlet

config
/WEB-INF/struts-config.xml


debug
3


detail
3

2



/WEB-INF/struts-html.tld
/WEB-INF/struts-html.tld



/WEB-INF/struts-logic.tld
/WEB-INF/struts-logic.tld




/WEB-INF/struts-nested.tld
/WEB-INF/struts-nested.tld




/WEB-INF/struts-template.tld
/WEB-INF/struts-template.tld

Struts1.1中提供了很詳細(xì)的例子,你可以仔細(xì)看看.
接下來你該根據(jù)需要配置struts-config.xml,以下是一個(gè)簡單的例子


"-//Apache Software Foundation//DTD Struts Configuration 1.1//EN"
"http://jakarta.apache.org/struts/dtds/struts-config_1_1.dtd">







type="com.bingo.finance.action.UserForm"/>












type="com.bingo.finance.action.UseregAction"
name="useregForm"
scope="request"
validate="true"
input="/usereg.jsp">






parameter="com.bingo.finance.common.DisplayMsg"/>










上面的英文我相信你能夠看懂。我就不做解釋了。你需要繼續(xù)配置validation.xml了,看如下
簡單的例子.



depends="required,mask,minlength,maxlength">




mask
^/w+$


minlength
5


maxlength
20


depends="required,mask,minlength,maxlength">




mask
^/w+$


minlength
5


maxlength
20


depends="required,mask,minlength,maxlength">




mask
^/w+$


minlength
5


maxlength
20


depends="required,mask,minlength,maxlength"> <
本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動(dòng)力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉