国产一级a片免费看高清,亚洲熟女中文字幕在线视频,黄三级高清在线播放,免费黄色视频在线看

打開(kāi)APP
userphoto
未登錄

開(kāi)通VIP,暢享免費(fèi)電子書(shū)等14項(xiàng)超值服

開(kāi)通VIP
教你如何一步一步深入Struts
 

 Struts介紹

 

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

1. 從 JSP 文件到 servlet 的流程

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

簡(jiǎn)單的獨(dú)立 JSP 文件
在小型 JSP 應(yīng)用程序中,經(jīng)常會(huì)看到數(shù)據(jù)、業(yè)務(wù)邏輯和用戶(hù)界面被組合在一個(gè)代碼模塊中。此外,應(yīng)用程序通常還包含用來(lái)控制應(yīng)用程序流程的邏輯。清單 1 和圖 2 展示了允許用戶(hù)加入一個(gè)郵件列表的一個(gè)簡(jiǎn)單 JSP 文件。

清單 1. join.jsp -- 一個(gè)簡(jiǎn)單的請(qǐng)求和響應(yīng) JSP 文件

<%@ page language="java" %>
<%@ page import="business.util.Validation" %>
<%@ page import="business.db.MailingList" %>
<%
String error = "";
String email = request.getParameter("email");
 
// 是否有電子郵件地址
if(      email!=null ) {
 
         // 驗(yàn)證輸入...
         if( business.util.Validation.isValidEmail(email) ) {
 
                 // 存儲(chǔ)輸入...
                 try {
                          business.db.MailingList.AddEmail(email);
                 } catch (Exception e) {
                          error = "Error adding email address to system. " + e;
                 }
 
                 if( error.length()==0 ) {
%>
                          // 重定向到歡迎頁(yè)...
                          <jsp:forward page="welcome.html"/>
<%
                 }
         } else {
                 // 設(shè)置錯(cuò)誤消息并重新顯示網(wǎng)頁(yè)
                 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. 在簡(jiǎn)單的請(qǐng)求和響應(yīng)中,JSP 文件設(shè)置數(shù)據(jù)、控制到下一個(gè)網(wǎng)頁(yè)的流程并創(chuàng)建 HTML
 

這個(gè)郵件列表 JSP 文件是一個(gè)獨(dú)立的、自主完成所有任務(wù)的模塊。未包含在這個(gè) JSP 文件中的僅有代碼是包含在 isValidEmail() 中的實(shí)際驗(yàn)證代碼和將電子郵件地址存入數(shù)據(jù)庫(kù)的代碼。(將 isValidEmail() 方法分離到可重用的代碼中似乎是當(dāng)然的選擇,但我曾見(jiàn)過(guò)直接嵌入網(wǎng)頁(yè)中的 isValidEmail() 代碼。單頁(yè)方法的優(yōu)點(diǎn)是易于理解,并且最初也易于構(gòu)建。此外,對(duì)于各種圖形化開(kāi)發(fā)工具,入門(mén)也很容易。

join.jsp 的活動(dòng)

  • 顯示打開(kāi)的輸入網(wǎng)頁(yè)。
  • 從表單參數(shù)中讀取 email 的值。
  • 驗(yàn)證 email 地址。
  • 如果 email 地址有效:
    • 將該地址添加到數(shù)據(jù)庫(kù)中。
    • 重定向到下一個(gè)網(wǎng)頁(yè)。
  • 如果 email 地址無(wú)效:
    • 設(shè)置錯(cuò)誤消息。
    • 重新顯示含有錯(cuò)誤消息的 join.jsp。

單頁(yè)方法的后果

  • HTML Java 強(qiáng)耦合在一起
    JSP
    文件的編寫(xiě)者必須既是網(wǎng)頁(yè)設(shè)計(jì)者,又是 Java 開(kāi)發(fā)者。其結(jié)果通常要么是很糟的 Java 代碼,要么是難看的網(wǎng)頁(yè),有時(shí)甚至 Java 代碼和網(wǎng)頁(yè)都很糟。
  • Java JavaScript 的不足
    隨著網(wǎng)頁(yè)逐漸變大,很容易想到實(shí)現(xiàn)一些 JavaScript。當(dāng)網(wǎng)頁(yè)中出現(xiàn) JavaScript 時(shí),這種腳本就可能與 Java 代碼產(chǎn)生混淆??赡墚a(chǎn)生混淆的一個(gè)例子是使用客戶(hù)端的 JavaScript 來(lái)驗(yàn)證 email 域。
  • 內(nèi)嵌的流程邏輯
    要理解應(yīng)用程序的整個(gè)流程,您必須瀏覽所有網(wǎng)頁(yè)。試想一下?lián)碛?/span> 100 個(gè)網(wǎng)頁(yè)的網(wǎng)站的錯(cuò)綜復(fù)雜的邏輯。
  • 調(diào)試?yán)щy
    除了很糟的外觀之外,HTML 標(biāo)記、Java 代碼和 JavaScript 代碼都集中在一個(gè)網(wǎng)頁(yè)中還使調(diào)試變得相當(dāng)困難。
  • 強(qiáng)耦合
    更改業(yè)務(wù)邏輯或數(shù)據(jù)可能牽涉相關(guān)的每個(gè)網(wǎng)頁(yè)。
  • 美學(xué)
    在很大的網(wǎng)頁(yè)中,這編碼樣式看起來(lái)雜亂無(wú)章。我過(guò)去進(jìn)行 Microsoft ASP 開(kāi)發(fā)時(shí),我經(jīng)??吹接?/span> 1000 行的網(wǎng)頁(yè)。即使有彩色語(yǔ)法顯示,閱讀和理解這些代碼仍然比較困難。

請(qǐng)別在我的 HTML 中加入太多的 Java 代碼
在清單 1 中,不是 Java 代碼中有大量的 HTML,而是在 HTML 文件中有大量的 Java 代碼。從這個(gè)觀點(diǎn)來(lái)看,除了允許網(wǎng)頁(yè)設(shè)計(jì)人員編寫(xiě) Java 代碼之外,我實(shí)際上沒(méi)做什么。但是,我們并不是一無(wú)所有;在 JSP 1.1 中,我們獲得一種稱(chēng)為“標(biāo)記”的新特性。

JSP 標(biāo)記只是將代碼從 JSP 文件中抽取出來(lái)的一種方式。有人將 JSP 標(biāo)記看作是 JSP 文件的宏,其中用于這個(gè)標(biāo)記的代碼包含在 servlet 中。(宏的觀點(diǎn)在很大程度上是正確的。)出于同樣的原因,我不希望在 Java 代碼中看到 HTML 標(biāo)記,我也不希望在 JSP 文件中看到 Java 代碼。JSP 技術(shù)的整個(gè)出發(fā)點(diǎn)就是允許網(wǎng)頁(yè)設(shè)計(jì)人員創(chuàng)建 servlet,而不必糾纏于 Java 代碼。標(biāo)記允許 Java 程序員將 Java 代碼偽裝成 HTML 來(lái)擴(kuò)展 JSP 文件。圖 3 顯示了從 JSP 網(wǎng)頁(yè)中抽取代碼并將它們放入 JSP 標(biāo)記中的一般概念。

3. JSP 標(biāo)記

清單 2 是用來(lái)說(shuō)明 Struts 標(biāo)記的功能的一個(gè)例子。在清單 2 中,正常的 HTML <form> 標(biāo)記被用 Struts <form:form> 標(biāo)記替換。清單 3 顯示了瀏覽器接收到的結(jié)果 HTML。瀏覽器獲得 HTML <form> 標(biāo)記,但帶有附加代碼,如 JavaScript。附加的 JavaScript 激活 email 地址域。服務(wù)器端的 <form:form> 標(biāo)記代碼創(chuàng)建適當(dāng)?shù)?HTML,并使網(wǎng)頁(yè)設(shè)計(jì)人員不再接觸 JavaScript。

清單 2. Struts 的 form 標(biāo)記

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

清單 3. 發(fā)送給瀏覽器的結(jié)果 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>
 

有關(guān) JSP 標(biāo)記的注意事項(xiàng):

  • JSP 標(biāo)記需要一個(gè)運(yùn)行 JSP 1.1 或更高版本的容器。
  • JSP 標(biāo)記在服務(wù)器上運(yùn)行,而不像 HTML 標(biāo)記那樣由客戶(hù)機(jī)解釋。
  • JSP 標(biāo)記提供了適當(dāng)?shù)拇a重用機(jī)制。
  • 可以使用一種稱(chēng)為 include JSP 機(jī)制將 HTML JavaScript 添加到網(wǎng)頁(yè)中。但是,開(kāi)發(fā)人員常常會(huì)創(chuàng)建巨大的 JavaScript 庫(kù)文件,這些庫(kù)文件被包含在 JSP 文件中。結(jié)果返回給客戶(hù)機(jī)的 HTML 網(wǎng)頁(yè)要比必需的 HMTL 網(wǎng)頁(yè)大得多。include 的正確用法是僅將它用于生成諸如頁(yè)眉和頁(yè)腳這類(lèi)內(nèi)容的 HTML 代碼段。
  • 通過(guò)抽取出 Java 代碼,JSP 標(biāo)記使開(kāi)發(fā)角色更加專(zhuān)業(yè)化。

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

  • Model(模型)
    模型包含應(yīng)用程序的核心功能。模型封裝了應(yīng)用程序的狀態(tài)。有時(shí)它包含的唯一功能就是狀態(tài)。它對(duì)視圖或控制器一無(wú)所知。
  • View(視圖)
    視圖提供模型的表示。它是應(yīng)用程序的外觀。視圖可以訪問(wèn)模型的讀方法,但不能訪問(wèn)寫(xiě)方法。此外,它對(duì)控制器一無(wú)所知。當(dāng)更改模型時(shí),視圖應(yīng)得到通知。
  • Controller(控制器)
    控制器對(duì)用戶(hù)的輸入作出反應(yīng)。它創(chuàng)建并設(shè)置模型。

MVC Model 2
Web 向軟件開(kāi)發(fā)人員提出了一些特有的挑戰(zhàn),最明顯的就是客戶(hù)機(jī)和服務(wù)器的無(wú)狀態(tài)連接。這種無(wú)狀態(tài)行為使得模型很難將更改通知視圖。在 Web 上,為了發(fā)現(xiàn)對(duì)應(yīng)用程序狀態(tài)的修改,瀏覽器必須重新查詢(xún)服務(wù)器。

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

  • Java 程序員應(yīng)該開(kāi)發(fā)服務(wù),而不是 HTML。
  • 更改布局時(shí)需要更改代碼。
  • 服務(wù)的用戶(hù)應(yīng)該能夠創(chuàng)建網(wǎng)頁(yè)來(lái)滿(mǎn)足它們的特定需要。
  • 網(wǎng)頁(yè)設(shè)計(jì)人員不能直接參與網(wǎng)頁(yè)開(kāi)發(fā)。
  • 嵌在代碼中的 HTML 很難看。

對(duì)于 Web,需要修改標(biāo)準(zhǔn)的 MVC 形式。圖 4 顯示了 MVC 的 Web 改寫(xiě)版,通常也稱(chēng)為 MVC Model 2 或 MVC 2。

4. MVC Model 2

Struts,MVC 2 的一種實(shí)現(xiàn)
Struts 是一組相互協(xié)作的類(lèi)、servlet 和 JSP 標(biāo)記,它們組成一個(gè)可重用的 MVC 2 設(shè)計(jì)。這個(gè)定義表示 Struts 是一個(gè)框架,而不是一個(gè)庫(kù),但 Struts 也包含了豐富的標(biāo)記庫(kù)和獨(dú)立于該框架工作的實(shí)用程序類(lèi)。圖 5 顯示了 Struts 的一個(gè)概覽。

5. Struts 概覽

Struts 概覽

  • Client browser(客戶(hù)瀏覽器)
    來(lái)自客戶(hù)瀏覽器的每個(gè) HTTP 請(qǐng)求創(chuàng)建一個(gè)事件。Web 容器將用一個(gè) HTTP 響應(yīng)作出響應(yīng)。
  • Controller(控制器)
    控制器接收來(lái)自瀏覽器的請(qǐng)求,并決定將這個(gè)請(qǐng)求發(fā)往何處。就 Struts 而言,控制器是以 servlet 實(shí)現(xiàn)的一個(gè)命令設(shè)計(jì)模式。struts-config.xml 文件配置控制器。
  • 業(yè)務(wù)邏輯
    業(yè)務(wù)邏輯更新模型的狀態(tài),并幫助控制應(yīng)用程序的流程。就 Struts 而言,這是通過(guò)作為實(shí)際業(yè)務(wù)邏輯包裝的 Action 類(lèi)完成的。
  • Model(模型)的狀態(tài)
    模型表示應(yīng)用程序的狀態(tài)。業(yè)務(wù)對(duì)象更新應(yīng)用程序的狀態(tài)。ActionForm bean 在會(huì)話級(jí)或請(qǐng)求級(jí)表示模型的狀態(tài),而不是在持久級(jí)。JSP 文件使用 JSP 標(biāo)記讀取來(lái)自 ActionForm bean 的信息。
  • View(視圖)
    視圖就是一個(gè) JSP 文件。其中沒(méi)有流程邏輯,沒(méi)有業(yè)務(wù)邏輯,也沒(méi)有模型信息 -- 只有標(biāo)記。標(biāo)記是使 Struts 有別于其他框架(如 Velocity)的因素之一。
詳細(xì)分析 Struts

圖 6 顯示的是
org.apache.struts.action 包的一個(gè)最簡(jiǎn) UML 圖。圖 6 顯示了 ActionServlet (Controller)、ActionForm (Form State) 和 Action (Model Wrapper) 之間的最簡(jiǎn)關(guān)系。

6. Command (ActionServlet) 與 Model (Action & ActionForm) 之間的關(guān)系的 UML 圖

ActionServlet 類(lèi)
您還記得函數(shù)映射的日子嗎?在那時(shí),您會(huì)將某些輸入事件映射到一個(gè)函數(shù)指針上。如果您對(duì)此比較熟悉,您會(huì)將配置信息放入一個(gè)文件,并在運(yùn)行時(shí)加載這個(gè)文件。函數(shù)指針數(shù)組曾經(jīng)是用 C 語(yǔ)言進(jìn)行結(jié)構(gòu)化編程的很好方法。

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

ActionServlet 是該 MVC 實(shí)現(xiàn)的 Command 部分,它是這一框架的核心。ActionServlet (Command) 創(chuàng)建并使用 Action、ActionFormActionForward。如前所述,struts-config.xml 文件配置該 Command。在創(chuàng)建 Web 項(xiàng)目時(shí),您將擴(kuò)展 ActionActionForm 來(lái)解決特定的問(wèn)題。文件 struts-config.xml 指示 ActionServlet 如何使用這些擴(kuò)展的類(lèi)。這種方法有幾個(gè)優(yōu)點(diǎn):

  • 應(yīng)用程序的整個(gè)邏輯流程都存儲(chǔ)在一個(gè)分層的文本文件中。這使得人們更容易查看和理解它,尤其是對(duì)于大型應(yīng)用程序而言。
  • 網(wǎng)頁(yè)設(shè)計(jì)人員不必費(fèi)力地閱讀 Java 代碼來(lái)理解應(yīng)用程序的流程。
  • Java 開(kāi)發(fā)人員也不必在更改流程以后重新編譯代碼。

可以通過(guò)擴(kuò)展 ActionServlet 來(lái)添加 Command 功能。

ActionForm 類(lèi)

ActionForm 維護(hù) Web 應(yīng)用程序的會(huì)話狀態(tài)。ActionForm 是一個(gè)抽象類(lèi),必須為每個(gè)輸入表單模型創(chuàng)建該類(lèi)的子類(lèi)。當(dāng)我說(shuō)輸入表單模型時(shí),是指 ActionForm 表示的是由 HTML 表單設(shè)置或更新的一般意義上的數(shù)據(jù)。例如,您可能有一個(gè)由 HTML 表單設(shè)置的 UserActionForm。Struts 框架將執(zhí)行以下操作:

  • 檢查 UserActionForm 是否存在;如果不存在,它將創(chuàng)建該類(lèi)的一個(gè)實(shí)例。
  • Struts 將使用 HttpServletRequest 中相應(yīng)的域設(shè)置 UserActionForm 的狀態(tài)。沒(méi)有太多討厭的 request.getParameter() 調(diào)用。例如,Struts 框架將從請(qǐng)求流中提取 fname,并調(diào)用 UserActionForm.setFname()。
  • Struts 框架在將 UserActionForm 傳遞給業(yè)務(wù)包裝 UserAction 之前將更新它的狀態(tài)。
  • 在將它傳遞給 Action 類(lèi)之前,Struts 還會(huì)對(duì) UserActionForm 調(diào)用 validation() 方法進(jìn)行表單狀態(tài)驗(yàn)證。注:這并不總是明智之舉。別的網(wǎng)頁(yè)或業(yè)務(wù)可能使用 UserActionForm,在這些地方,驗(yàn)證可能有所不同。在 UserAction 類(lèi)中進(jìn)行狀態(tài)驗(yàn)證可能更好。
  • 可在會(huì)話級(jí)維護(hù) UserActionForm

注:

  • struts-config.xml 文件控制 HTML 表單請(qǐng)求與 ActionForm 之間的映射關(guān)系。
  • 可將多個(gè)請(qǐng)求映射到 UserActionForm
  • UserActionForm 可跨多頁(yè)進(jìn)行映射,以執(zhí)行諸如向?qū)е?lèi)的操作。

Action 類(lèi)
Action 類(lèi)是業(yè)務(wù)邏輯的一個(gè)包裝。Action 類(lèi)的用途是將 HttpServletRequest 轉(zhuǎn)換為業(yè)務(wù)邏輯。要使用 Action,請(qǐng)創(chuàng)建它的子類(lèi)并覆蓋 process() 方法。

ActionServlet (Command) 使用 perform() 方法將參數(shù)化的類(lèi)傳遞給 ActionForm。仍然沒(méi)有太多討厭的 request.getParameter() 調(diào)用。當(dāng)事件進(jìn)展到這一步時(shí),輸入表單數(shù)據(jù)(或 HTML 表單數(shù)據(jù))已被從請(qǐng)求流中提取出來(lái)并轉(zhuǎn)移到 ActionForm 類(lèi)中。

注:擴(kuò)展 Action 類(lèi)時(shí)請(qǐng)注意簡(jiǎn)潔。Action 類(lèi)應(yīng)該控制應(yīng)用程序的流程,而不應(yīng)該控制應(yīng)用程序的邏輯。通過(guò)將業(yè)務(wù)邏輯放在單獨(dú)的包或 EJB 中,我們就可以提供更大的靈活性和可重用性。

考慮 Action 類(lèi)的另一種方式是 Adapter 設(shè)計(jì)模式。Action 的用途是“將類(lèi)的接口轉(zhuǎn)換為客戶(hù)機(jī)所需的另一個(gè)接口。Adapter 使類(lèi)能夠協(xié)同工作,如果沒(méi)有 Adapter,則這些類(lèi)會(huì)因?yàn)椴患嫒莸慕涌诙鵁o(wú)法協(xié)同工作。”(摘自 Gof 所著的 Design Patterns - Elements of Reusable OO Software)。本例中的客戶(hù)機(jī)是 ActionServlet,它對(duì)我們的具體業(yè)務(wù)類(lèi)接口一無(wú)所知。因此,Struts 提供了它能夠理解的一個(gè)業(yè)務(wù)接口,即 Action。通過(guò)擴(kuò)展 Action,我們使得我們的業(yè)務(wù)接口與 Struts 業(yè)務(wù)接口保持兼容。(一個(gè)有趣的發(fā)現(xiàn)是, Action 是類(lèi)而不是接口)。Action 開(kāi)始為一個(gè)接口,后來(lái)卻變成了一個(gè)類(lèi)。真是金無(wú)足赤。)

Error 類(lèi)
UML 圖(圖 6)還包括
ActionErrorActionErrors。ActionError 封裝了單個(gè)錯(cuò)誤消息。ActionErrorsActionError 類(lèi)的容器,View 可以使用標(biāo)記訪問(wèn)這些類(lèi)。ActionError 是 Struts 保持錯(cuò)誤列表的方式。

7. Command (ActionServlet) 與 Model (Action) 之間的關(guān)系的 UML 圖

ActionMapping 類(lèi)
輸入事件通常是在 HTTP 請(qǐng)求表單中發(fā)生的,servlet 容器將 HTTP 請(qǐng)求轉(zhuǎn)換為
HttpServletRequest??刂破鞑榭摧斎胧录⒄?qǐng)求分派給某個(gè) Action 類(lèi)。struts-config.xml 確定 Controller 調(diào)用哪個(gè) Action 類(lèi)。struts-config.xml 配置信息被轉(zhuǎn)換為一組 ActionMapping,而后者又被放入 ActionMappings 容器中。(您可能尚未注意到這一點(diǎn),以 s 結(jié)尾的類(lèi)就是容器)

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

ActionMappings
ActionMappingsActionMapping 對(duì)象的一個(gè)集合。

再訪郵件列表樣例
下面我們看一下 Struts 是如何解決困擾
join.jsp 的這些問(wèn)題的。改寫(xiě)后的方案由兩個(gè)項(xiàng)目組成。第一個(gè)項(xiàng)目包含應(yīng)用程序的邏輯部分,這個(gè)應(yīng)用程序是獨(dú)立于 Web 應(yīng)用程序的。這個(gè)獨(dú)立層可能是用 EJB 技術(shù)實(shí)現(xiàn)的公共服務(wù)層。為了便于說(shuō)明,我使用 Ant 構(gòu)建進(jìn)程創(chuàng)建了一個(gè)稱(chēng)為 business 的包。有幾個(gè)原因促使我們使用獨(dú)立的業(yè)務(wù)層:

  • 劃分責(zé)任
    單獨(dú)的包使管理人員能夠在開(kāi)發(fā)小組內(nèi)委派責(zé)任。這也有助于提高開(kāi)發(fā)人員的責(zé)任心。
  • 通用件
    我們?cè)O(shè)想開(kāi)發(fā)人員將這個(gè)包看作一個(gè)商業(yè)軟件。將它放在另外的包中使它更像通用件。這個(gè)包可能是通用件,也可能是由組織內(nèi)部的另一個(gè)小組開(kāi)發(fā)的。
  • 避免不必要的構(gòu)建和單元測(cè)試。
    分開(kāi)的構(gòu)建進(jìn)程有助于避免不必要的構(gòu)建和單元測(cè)試。
  • 使用接口開(kāi)發(fā)
    在進(jìn)行開(kāi)發(fā)和避免不必要的耦合時(shí),它有助于從接口的觀點(diǎn)來(lái)思考問(wèn)題。這是極重要的一個(gè)方面。當(dāng)開(kāi)發(fā)您自己的業(yè)務(wù)包時(shí),這些業(yè)務(wù)類(lèi)不應(yīng)該關(guān)心到底是 Web 應(yīng)用程序執(zhí)行調(diào)用,還是獨(dú)立應(yīng)用程序執(zhí)行調(diào)用。因此,應(yīng)該避免在業(yè)務(wù)邏輯層使用對(duì) servlet API Struts API 調(diào)用的任何引用。
  • 穩(wěn)定性
    并不是每個(gè)組織都每天、每周甚至每月進(jìn)行檢修。因此,在進(jìn)行開(kāi)發(fā)時(shí),穩(wěn)定的接口點(diǎn)是重要的。不能因?yàn)闃I(yè)務(wù)包處于變遷階段就認(rèn)為 Web 項(xiàng)目也應(yīng)該處于變遷階段。

業(yè)務(wù)構(gòu)建注釋
我用 Ant 構(gòu)建項(xiàng)目,并用 JUnit 運(yùn)行單元測(cè)試。business.zip 包含構(gòu)建業(yè)務(wù)項(xiàng)目所需的一切,當(dāng)然 Ant 和 JUnit 除外。這個(gè)包腳本將構(gòu)建類(lèi),運(yùn)行單元測(cè)試,創(chuàng)建 Java 文檔和 jar 文件,最后將所有這些內(nèi)容壓縮到一個(gè) zip 文件中發(fā)送給客戶(hù)。只要對(duì)
build.xml 作一些修改,您就可以將它部署到其他平臺(tái)上。Business.jar 位于 Web 的下載部分,因此,您并非必須下載并構(gòu)建這個(gè)業(yè)務(wù)包。

Web 項(xiàng)目
第二個(gè)項(xiàng)目是用 Struts 開(kāi)發(fā)的一個(gè) Web 應(yīng)用程序。您將需要一個(gè)符合 JSP 1.1 和 Servlet 2.2 規(guī)范的容器。最快的入門(mén)方法是下載并安裝 Tomcat 3.2(請(qǐng)參閱參考資源)。直到有 Struts 的 1.0 發(fā)行版之前,我建議您從 Jakarta 項(xiàng)目獲得最新的版本(請(qǐng)參閱參考資源)。這對(duì)我來(lái)說(shuō)是個(gè)大問(wèn)題,我不能確保我的 Web 項(xiàng)目樣例能與您下載的 Struts 一起工作。Struts 仍在不斷變化,所以我不得不經(jīng)常更新我的項(xiàng)目。在本項(xiàng)目中,我使用的是 jakarta-struts-20010105.zip。圖 8 顯示了此 Web 項(xiàng)目的結(jié)構(gòu)。如果您已安裝了 Ant,則運(yùn)行這個(gè)版本將創(chuàng)建一個(gè)稱(chēng)為
joinStruts.war 的 war 文件,您隨時(shí)可以部署這個(gè)文件。

8. Web 項(xiàng)目的結(jié)構(gòu)

清單 4 顯示了轉(zhuǎn)換后的 JSP 文件,稱(chēng)為 joinMVC.jsp。這個(gè)文件從最初的 50 行變?yōu)?19 行,并且現(xiàn)在不含任何 Java 代碼。從網(wǎng)頁(yè)設(shè)計(jì)人員的角度來(lái)看,這是個(gè)巨大的改進(jìn)。

清單 4. joinMVC.jsp -- 再訪簡(jiǎn)單的 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>

網(wǎng)頁(yè)的變化
下面是使用 Struts 標(biāo)記庫(kù)之后所發(fā)生變化的列表:

  • Import
<%@ taglib uri="/WEB-INF/struts.tld" prefix="struts" %>
  • 用于 Java 代碼的 <%@page import? 已被替換為用于 Struts 標(biāo)記庫(kù)的 <%@ taglib uri?
  • 文本
<struts:message key="join.title"/>
  • 資源屬性文件包含 join.title 的文本。在本例中,ApplicationResources 屬性文件包含這個(gè)名值對(duì)。這使字符串更易于查看和國(guó)際化。
  • 錯(cuò)誤
<form:errors/>
  • ActionServlet ActionForm 構(gòu)建要顯示的錯(cuò)誤消息。這些錯(cuò)誤消息也可以包含在屬性文件中。ApplicationResources 也提供了一種格式化錯(cuò)誤的方法,即設(shè)置 error.header error.footer。
  • HTML 表單
<form:form action="join.do" focus="email" >
    • JSP <form> 標(biāo)記和屬性替代了 HTML <form> 標(biāo)記和屬性。 <form action="join.jsp" name="join"> 已更改為 <form:form action="join.do" focus="email" >。
    • HTML <input> 標(biāo)記已替換為 <form:text/>。
    • HTML <submit> 標(biāo)記已替換為 <form:submit/>。

模型 -- 會(huì)話狀態(tài)
JoinForm 擴(kuò)展了 ActionForm 并包含表單數(shù)據(jù)。本例中的表單數(shù)據(jù)只有電子郵件地址。我已為電子郵件地址添加了一個(gè)寫(xiě)方法和讀方法,以供框架訪問(wèn)。為了便于說(shuō)明,我重寫(xiě)了 validate() 方法,并使用了 Struts 的跟蹤功能。Struts 將創(chuàng)建 JoinForm 并設(shè)置狀態(tài)信息。

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

清單 5. - JoinAction.perform()

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

注:perform() 返回一個(gè)稱(chēng)為 ActionForward 的類(lèi),該類(lèi)通知控制器下一步該執(zhí)行什么操作。在本例中,我使用從控制器傳入的映射來(lái)決定下一步的操作。

控制器
我已修改了 JSP 文件,并創(chuàng)建了兩個(gè)新類(lèi):一個(gè)類(lèi)用來(lái)包含表單數(shù)據(jù),一個(gè)類(lèi)用來(lái)調(diào)用業(yè)務(wù)包。最后,我通過(guò)修改配置文件
struts-config.xml 將它們整合起來(lái)。清單 6 顯示了我添加的 action 元素,這個(gè)元素用來(lái)控制 joinMVC.jsp 的流程。

清單 6. Action 配置

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

action 元素描述了從請(qǐng)求路徑到相應(yīng)的 Action 類(lèi)的映射,應(yīng)該用這些類(lèi)來(lái)處理來(lái)自這個(gè)路徑的請(qǐng)求。每個(gè)請(qǐng)求類(lèi)型都應(yīng)該有相應(yīng)的 action 元素,用來(lái)描述如何處理該請(qǐng)求。對(duì)于 join 請(qǐng)求:

  • joinForm 用來(lái)容納表單數(shù)據(jù)。
  • 因?yàn)?/span> validate 被標(biāo)記為 true,所以 joinForm 將試圖進(jìn)行自我驗(yàn)證。
  • web.mailinglist.JoinAction 是用來(lái)處理對(duì)這個(gè)映射的請(qǐng)求的 action 類(lèi)。
  • 如果一切順利,該請(qǐng)求將轉(zhuǎn)到 welcome.jsp。
  • 如果出現(xiàn)業(yè)務(wù)邏輯故障,流程將返回到 joinMVC.jsp,這是最初發(fā)出請(qǐng)求的網(wǎng)頁(yè)。為什么會(huì)這樣呢?在清單 6 action 元素中,有一個(gè)稱(chēng)為 input 的屬性,其值為 "/joinMVC.jsp"。在我的 JoinAction.perform()(如清單 5 所示)中,如果業(yè)務(wù)邏輯失敗,perform() 就返回一個(gè) ActionForward,并以 mapping.getInput() 作為參數(shù)。本例中的 getInput() "/joinMVC.jsp"。如果業(yè)務(wù)邏輯失敗,它將返回到 joinMVC.jsp,這是最初發(fā)出請(qǐng)求的網(wǎng)頁(yè)。

使用 Struts 前后的比較
正如我們?cè)趫D 9 中所看到的那樣,復(fù)雜性和層都有顯著增加。不再存在從 JSP 文件到 Service 層的直接調(diào)用。

9. 使用 Struts 前后的比較

Struts 的優(yōu)點(diǎn)

  • JSP 標(biāo)記機(jī)制的使用
    標(biāo)記特性從 JSP 文件獲得可重用代碼和抽象 Java 代碼。這個(gè)特性能很好地集成到基于 JSP 的開(kāi)發(fā)工具中,這些工具允許用標(biāo)記編寫(xiě)代碼。
  • 標(biāo)記庫(kù)
    為什么要另發(fā)明一種輪子,或標(biāo)記庫(kù)呢?如果您在庫(kù)中找不到您所要的標(biāo)記,那就自己定義吧。此外,如果您正在學(xué)習(xí) JSP 標(biāo)記技術(shù),則 Struts 為您提供了一個(gè)起點(diǎn)。
  • 開(kāi)放源碼
    您可以獲得開(kāi)放源碼的全部?jī)?yōu)點(diǎn),比如可以查看代碼并讓使用庫(kù)的每個(gè)人檢查代碼。許多人都可以進(jìn)行很好的代碼檢查。
  • MVC 實(shí)現(xiàn)樣例
    如果您希望創(chuàng)建您自己的 MVC 實(shí)現(xiàn),則 Struts 可增加您的見(jiàn)識(shí)。
  • 管理問(wèn)題空間
    分治是解決問(wèn)題并使問(wèn)題可管理的極好方法。當(dāng)然,這是一把雙刃劍。問(wèn)題越來(lái)越復(fù)雜,并且需要越來(lái)越多的管理。

Struts 的缺點(diǎn)

  • 仍處于發(fā)展初期
    Struts
    開(kāi)發(fā)仍處于初級(jí)階段。他們正在向著發(fā)行版本 1.0 而努力,但與任何 1.0 版本一樣,它不可能盡善盡美。
  • 仍在變化中
    這個(gè)框架仍在快速變化。Struts 1.0 Struts 0.5 相比變化極大。為了避免使用不贊成使用的方法,您可能隔一天就需要下載最新的 Struts。在過(guò)去的 6 個(gè)月中,我目睹 Struts 庫(kù)從 90K 增大到 270K 以上。由于 Struts 中的變化,我不得不數(shù)次修改我的示例,但我不保證我的示例能與您下載的 Struts 協(xié)同工作。
  • 正確的抽象級(jí)別
    Struts
    是否提供了正確的抽象級(jí)別?對(duì)于網(wǎng)頁(yè)設(shè)計(jì)人員而言,什么是正確的抽象級(jí)別呢?這是一個(gè)用 $64K 的文字才能解釋清楚的問(wèn)題。在開(kāi)發(fā)網(wǎng)頁(yè)的過(guò)程中,我們是否應(yīng)該讓網(wǎng)頁(yè)設(shè)計(jì)人員訪問(wèn) Java 代碼?某些框架(如 Velocity)說(shuō)不應(yīng)該,但它提供了另一種 Web 開(kāi)發(fā)語(yǔ)言讓我們學(xué)習(xí)。在 UI 開(kāi)發(fā)中限制訪問(wèn) Java 有一定的合理性。最重要的是,如果讓網(wǎng)頁(yè)設(shè)計(jì)人員使用一點(diǎn) Java,他將使用大量的 Java。在 Microsoft ASP 的開(kāi)發(fā)中,我總是看到這樣的情況。在 ASP 開(kāi)發(fā)中,您應(yīng)該創(chuàng)建 COM 對(duì)象,然后編寫(xiě)少量的 ASP 腳本將這些 COM 對(duì)象聯(lián)系起來(lái)。但是,ASP 開(kāi)發(fā)人員會(huì)瘋狂地使用 ASP 腳本。我會(huì)聽(tīng)到這樣的話,既然我可以用 VBScript 直接編寫(xiě) COM 對(duì)象,為什么還要等 COM 開(kāi)發(fā)人員來(lái)創(chuàng)建它呢?通過(guò)使用標(biāo)記庫(kù),Struts 有助于限制 JSP 文件中所需的 Java 代碼的數(shù)量。Logic Tag 就是這樣的一種庫(kù),它對(duì)有條件地生成輸出進(jìn)行管理,但這并不能阻止 UI 開(kāi)發(fā)人員對(duì) Java 代碼的狂熱。無(wú)論您決定使用哪種類(lèi)型的框架,您都應(yīng)該了解您要在其中部署和維護(hù)該框架的環(huán)境。當(dāng)然,這項(xiàng)任務(wù)真是說(shuō)起來(lái)容易做起來(lái)難。
  • 有限的適用范圍
    Struts
    是一種基于 Web MVC 解決方案,所以必須用 HTMLJSP 文件和 servlet 來(lái)實(shí)現(xiàn)它。
  • J2EE 應(yīng)用程序支持
    Struts
    需要支持 JSP 1.1 Servlet 2.2 規(guī)范的 servlet 容器。僅憑這一點(diǎn)遠(yuǎn)不能解決您的全部安裝問(wèn)題,除非使用 Tomcat 3.2。我用 Netscape iPlanet 6.0 安裝這個(gè)庫(kù)時(shí)遇到一大堆問(wèn)題,按理說(shuō)它是第一種符合 J2EE 的應(yīng)用程序服務(wù)器。我建議您在遇到問(wèn)題時(shí)訪問(wèn) Struts 用戶(hù)郵件列表的歸檔資料(請(qǐng)參閱參考資源)。
  • 復(fù)雜性
    在將問(wèn)題分為幾個(gè)部分的同時(shí)也引入了復(fù)雜性。毫無(wú)疑問(wèn),要理解 Struts 必須接受一定的培訓(xùn)。隨著變化的不斷加入,這有時(shí)會(huì)令人很沮喪。歡迎訪問(wèn)本網(wǎng)站。
  • 在何處...
    我還能指出其他問(wèn)題,例如,控制器的客戶(hù)端驗(yàn)證、可適用工作流程和動(dòng)態(tài)策略模式在什么地方?但是,目前這太容易成為吹毛求疵的問(wèn)題,有些問(wèn)題是無(wú)關(guān)緊要的,或者說(shuō)應(yīng)該對(duì) 1.0 發(fā)行版提這些問(wèn)題。隨著 Struts 小組的不斷努力,到您閱讀本文時(shí) Struts 說(shuō)不定已經(jīng)有了這些功能,或者它很快就會(huì)具有這些功能。

Struts 的前景
在這個(gè)軟件開(kāi)發(fā)的新時(shí)代,一切都變得很快。在不到 5 年的時(shí)間內(nèi),我已經(jīng)目睹了從 cgi/perl 到 ISAPI/NSAPI、再到使用 VB 的 ASP、一直到現(xiàn)在的 Java 和 J2EE 的變遷。Sun 正在盡力將新的變化反映到 JSP/servlet 體系結(jié)構(gòu)中,正如他們對(duì) Java 語(yǔ)言和 API 所作的更改一樣。您可以從 Sun 的網(wǎng)站獲得新的 JSP 1.2 和 Servlet 2.3 規(guī)范的草案。此外,一個(gè)標(biāo)準(zhǔn) JSP 標(biāo)記庫(kù)即將出現(xiàn);有關(guān)這些規(guī)范和標(biāo)記庫(kù)的鏈接,請(qǐng)參閱參考資源。

最后的注釋
Struts 使用標(biāo)記和 MVC 解決了某些重大問(wèn)題。這個(gè)方法有助于提高代碼的可重用性和靈活性。通過(guò)將問(wèn)題劃分為更小的組件,當(dāng)技術(shù)空間或問(wèn)題空間中出現(xiàn)變化時(shí),您就有更多的機(jī)會(huì)重用代碼。此外,Struts 使網(wǎng)頁(yè)設(shè)計(jì)人員和 Java 開(kāi)發(fā)人員能將精力集中于自己最擅長(zhǎng)的方面。但是,在強(qiáng)健性增強(qiáng)的同時(shí),也意味著復(fù)雜性的增加。Struts 比簡(jiǎn)單的單個(gè) JSP 網(wǎng)頁(yè)要復(fù)雜得多,但對(duì)于更大的系統(tǒng)而言,Struts 實(shí)際上有助于管理復(fù)雜性。另外,我并不想編寫(xiě)自己的 MVC 實(shí)現(xiàn),而只想了解一個(gè)這樣的實(shí)現(xiàn)。不管您是否會(huì)使用 Struts,回顧這個(gè) Struts 框架(對(duì)不起,應(yīng)該是庫(kù))都會(huì)使您對(duì) JSP 文件和 servlet 的特性、以及如何將它們組合起來(lái)用于您的下一個(gè) Web 項(xiàng)目有更好的了解。正像翼間支柱是機(jī)翼結(jié)構(gòu)中不可缺少的一部分一樣,Strut 也可能成為您下一個(gè) Web 項(xiàng)目的不可缺少的一部分。

參考資源

作者簡(jiǎn)介
Malcolm G. Davis 住在阿拉巴馬州伯明翰市,他在自己的咨詢(xún)公司當(dāng)總裁。他自稱(chēng)是一名 Java 傳道者。他在宣傳 Java 的優(yōu)點(diǎn)的閑暇之余,他會(huì)去長(zhǎng)跑,或者與自己的孩子一起玩??梢酝ㄟ^(guò) malcolm@nuearth.com 與 Malcolm 聯(lián)系。

 

本站僅提供存儲(chǔ)服務(wù),所有內(nèi)容均由用戶(hù)發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)點(diǎn)擊舉報(bào)。
打開(kāi)APP,閱讀全文并永久保存 查看更多類(lèi)似文章
猜你喜歡
類(lèi)似文章
Struts,MVC 的一種開(kāi)放源碼實(shí)現(xiàn)
開(kāi)源框架:Structs2.0入門(mén)學(xué)習(xí)(1)
【使用 struts portlet 在門(mén)戶(hù)應(yīng)用程序中實(shí)現(xiàn)頁(yè)面導(dǎo)航】
重用標(biāo)題,簡(jiǎn)化用戶(hù)界面
用struts framework開(kāi)發(fā)應(yīng)用程序
Struts標(biāo)記庫(kù)
更多類(lèi)似文章 >>
生活服務(wù)
分享 收藏 導(dǎo)長(zhǎng)圖 關(guān)注 下載文章
綁定賬號(hào)成功
后續(xù)可登錄賬號(hào)暢享VIP特權(quán)!
如果VIP功能使用有故障,
可點(diǎn)擊這里聯(lián)系客服!

聯(lián)系客服