在Succeeding with Struts的前面安裝部分,我間接提到了DynaForms在運(yùn)行期內(nèi)可以動(dòng)態(tài)的控制表格大小。換句話說(shuō),就是能夠根據(jù)需要得到5行、或者10行、或者15行長(zhǎng)的表格。可能有點(diǎn)不明智,我把這種策略的實(shí)際實(shí)現(xiàn)作為一種練習(xí)留給了讀者自己。在接下來(lái)的幾個(gè)月內(nèi),我收到了幾十個(gè)讀者的請(qǐng)求,他們請(qǐng)求給出詳細(xì)的實(shí)現(xiàn)細(xì)節(jié),所以這個(gè)月我將用兩種不同的方法來(lái)實(shí)現(xiàn)動(dòng)態(tài)調(diào)整的表格。 第一個(gè)方法就是我在前面的欄目中提到的那個(gè)方法,將尺寸參數(shù)留給DynaForm 的form-property 屬性來(lái)實(shí)現(xiàn)。為了演示詳細(xì)過(guò)程,我們來(lái)看看一個(gè)非常簡(jiǎn)單的應(yīng)用:添加關(guān)于不同Star Wars 演員的注釋。在這個(gè)應(yīng)用中我們感興趣的關(guān)鍵事實(shí)是:演員的數(shù)量在表格配置中動(dòng)態(tài)設(shè)定,而不是在struts-config.xml文件中動(dòng)態(tài)設(shè)定。 首先,我們先來(lái)看看struts-config.xml 文件:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts-config PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 1.1//EN"
"http://jakarta.apache.org/struts/dtds/struts-config_1_1.dtd">
<struts-config>
<form-beans>
<form-bean name="dynamicArrayForm" type="org.apache.struts.validator.DynaValidatorForm">
<form-property name="people" type="demo.Person[]"/>
</form-bean>
</form-beans>
<action-mappings>
<action path="/setupForm" type="demo.SetupFormAction" name="dynamicArrayForm" scope="session"
validate="false">
<forward name="success" path="/displayForm.jsp"/>
</action>
<action path="/processActorComments"
type="demo.ProcessFormAction"
name="dynamicArrayForm" scope="session"
validate="false">
<forward name="success" path="/displayForm.jsp"/>
</action>
</action-mappings>
</struts-config>
如你所見(jiàn),這是一個(gè)相當(dāng)簡(jiǎn)單的配置文件,只定義了一個(gè)表格和兩個(gè)動(dòng)作。第一個(gè)動(dòng)作,/setupForm,用來(lái)在初始顯示之前配置表格;另一個(gè)動(dòng)作,/processActorComments 用來(lái)處理用戶輸入的注釋。 在這個(gè)文件中有兩個(gè)重要的事情需要注意,它們對(duì)于事態(tài)的發(fā)展很關(guān)鍵: 1. people 表格屬性定義為demo.Person[] 類型(即demo.Person的一個(gè)排列),但不給出任何size 參數(shù)。這就為要?jiǎng)?chuàng)建的排列產(chǎn)生了一個(gè)占位符,但是沒(méi)有任何例示的實(shí)排列。 2. 這兩個(gè)動(dòng)作將表格定義在會(huì)話期范圍內(nèi)。這是很關(guān)鍵的,因?yàn)橛脩粼谔顚?xiě)數(shù)值之后提交表格時(shí),數(shù)值在動(dòng)作執(zhí)行之前已經(jīng)填充到表格內(nèi)了。這就意味著沒(méi)有機(jī)會(huì)手動(dòng)創(chuàng)建具有恰當(dāng)空位數(shù)的排列,正如你在表格顯示之前在SetupFormAction 類中看到的情況一樣。換句話說(shuō),當(dāng)表格提交時(shí),必須已經(jīng)有恰當(dāng)?shù)目瘴粊?lái)接受表格值,唯一能保證這個(gè)的方法就是在會(huì)話期范圍內(nèi)就已經(jīng)有了這個(gè)表格。 基本上在Person bean 中是沒(méi)有值的,他只是一個(gè)具有l(wèi)astName、 firstName、 dateOfBirth、gender 和comment字段的普通bean。源文件包括在WAR 文件內(nèi)。 現(xiàn)在我們來(lái)看看SetupFormAction 類,它在表格第一次顯示之前調(diào)用。
package demo;
/**
* Copyright 2004, James M. Turner.
* All Rights Reserved
*
* A Struts action that sets up a DynaForm which is globally scoped
*/
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.*;
import org.apache.struts.action.*;
import org.apache.struts.validator.DynaValidatorForm;
public class SetupFormAction extends Action {
public ActionForward execute(ActionMapping mapping, ActionForm form, HttpServletRequest request,
HttpServletResponse response)
throws ServletException, IOException {
DynaValidatorForm df = (DynaValidatorForm) form;
Person[] p = new Person[3];
p[0] = new Person();
p[0].setDateOfBirth("07/13/1942");
p[0].setLastName("Ford");
p[0].setFirstName("Harrison");
p[0].setGender("M");
p[1] = new Person();
p[1].setDateOfBirth("10/21/1956");
p[1].setLastName("Fisher");
p[1].setFirstName("Carrie");
p[1].setGender("F");
p[2] = new Person();
p[2].setDateOfBirth("09/25/1951");
p[2].setLastName("Hamill");
p[2].setFirstName("Mark");
p[2].setGender("M");
df.set("people", p);
return mapping.findForward("success");
}
}
這一次也沒(méi)有許多東西要看的。execute 方法要做的第一件事情,和任何基于DynaForm的動(dòng)作所做的一樣,就是將泛型ActionForm 類放到DynaValidatorForm內(nèi)。這就使得我們可以在表格上使用get和set 方法。第二件事情就是,創(chuàng)建一個(gè)具有三個(gè)元素的類型Person 的排列。在這個(gè)方法中,尺寸是硬布線的,在實(shí)際應(yīng)用中可以從數(shù)據(jù)庫(kù)中選擇一個(gè)尺寸。我們需要考慮的重要事情是排列應(yīng)該在代碼中創(chuàng)建,而不是由Struts引擎自己創(chuàng)建。這樣行數(shù)可根據(jù)應(yīng)用要求由代碼隨意指定。 一旦排列已經(jīng)確定,方法將創(chuàng)建三個(gè)Person 類實(shí)例并賦與數(shù)值。同樣,在實(shí)際的應(yīng)用中可通過(guò)一個(gè)循環(huán)來(lái)實(shí)現(xiàn),這個(gè)循環(huán)不斷地從數(shù)據(jù)庫(kù)中讀取行和填充表格行。最后,動(dòng)作返回成功,導(dǎo)致Struts轉(zhuǎn)移控制到displayForm.jsp 頁(yè)。
<!--
Copyright 2004, James M Turner.
All Rights Reserved
-->
<%@ taglib uri="/WEB-INF/struts-html.tld" prefix="html" %>
<%@ taglib uri="/WEB-INF/c.tld" prefix="c" %>
<head>
<title>Star Wars Actor Fact Page</title>
</head>
<H1><center>Start Wars Actor Fact Page</title>
<html:form action="/processActorComments" >
<table border="1" width="80%">
<tr><th>Last Name</th><th>First Name</th><th>Date of Birth</th><th>Comment</th></tr>
<c:forEach var="people" items="${dynamicArrayForm.map.people}">
<tr><td><c:out value="${people.lastName}"/></td>
<td><c:out value="${people.firstName}"/></td>
<td><c:out value="${people.dateOfBirth}"/></td>
<td><html:text name="people" indexed="true" property="comment"/></td>
</tr>
</c:forEach>
</table>
<P/>
<html:submit value="Update Comments"/>
</html:form>
同樣,這里也沒(méi)有很多東西要看的,他與我們上一篇文章查看固定長(zhǎng)度的行時(shí)的代碼完全一樣。該頁(yè)迭代行(記住在JSTL中我們必須使用map 屬性來(lái)獲得到DynaForm 屬性的訪問(wèn)),顯示演員的姓、名和出生日期,并提供文本域以便輸入注釋。 當(dāng)我們聚焦我們的瀏覽器合請(qǐng)求時(shí),http://localhost:8080/struts/setupForm.do (假設(shè)你把struts.war 文件放在你本地機(jī)器的Tomcat 內(nèi)),將會(huì)出現(xiàn)下列頁(yè)面:
Start Wars Actor Fact Page
一旦表格提供,另一個(gè)簡(jiǎn)單的Struts動(dòng)作來(lái)處理結(jié)果:
package demo;
/**
* Copyright 2004, James M. Turner.
* All Rights Reserved
*
* A Struts action that sends the new comments to the console
*/
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.*;
import org.apache.struts.action.*;
import org.apache.struts.validator.DynaValidatorForm;
public class ProcessFormAction extends Action {
public ActionForward execute(ActionMapping mapping,
ActionForm form,
HttpServletRequest request,
HttpServletResponse response)
throws ServletException, IOException {
DynaValidatorForm df = (DynaValidatorForm) form;
Person[] p = (Person[]) df.get("people");
for (int i = 0; i < p.length; i++) {
System.out.println(p[i].getFirstName() + " " + p[i].
getLastName() + ":" + p[i].getComment());
}
return mapping.findForward("success");
}
}
在實(shí)際的應(yīng)用中,這就是數(shù)據(jù)寫(xiě)回到數(shù)據(jù)庫(kù)的地方。在這種情況下,他只將數(shù)據(jù)倒在控制臺(tái)上所以我們可以看到他是正確收到的。假設(shè)我們?yōu)槊總€(gè)演員都填充了恰當(dāng)?shù)闹担覀冊(cè)诳刂婆_(tái)上會(huì)看到下列內(nèi)容:
Harrison Ford:Indiana Jones
Carrie Fisher:Postcards from the Edge
Mark Hamill:Wing Commander
正如我在文章開(kāi)頭提到的一樣,還有另一個(gè)方法可以解決這個(gè)問(wèn)題,而且它不需要使用會(huì)話期范圍內(nèi)的表格。這個(gè)方法就是使用HashMaps 來(lái)存儲(chǔ)行。我們來(lái)看看使用HashMaps編寫(xiě)的同一段代碼:
首先,我們添加一個(gè)新表格到struts-config.xml:
<form-bean name="dynamicHashmapForm" type="org.apache.struts.validator.DynaValidatorForm">
<form-property name="people" type="java.util.HashMap"/>
<form-property name="comments" type="java.util.HashMap"/>
</form-bean>
現(xiàn)在,我們不使用beans的排列,改為使用HashMap 來(lái)存儲(chǔ)每個(gè)人的數(shù)據(jù)。另外,我們需要一個(gè)新的HashMap 來(lái)存儲(chǔ)注釋,原因我稍后再解釋。我們也需要一個(gè)新的動(dòng)作來(lái)填充數(shù)據(jù):
package demo;
/**
* Copyright 2004, James M. Turner.
* All Rights Reserved
*
* A Struts action that sets up a DynaForm which is globally scoped
*/
import java.io.IOException;
import java.util.HashMap;
import javax.servlet.ServletException;
import javax.servlet.http.*;
import org.apache.struts.action.*;
import org.apache.struts.validator.DynaValidatorForm;
public class SetupHashFormAction extends Action {
public ActionForward execute(ActionMapping mapping,
ActionForm form, HttpServletRequest request,
HttpServletResponse response)
throws ServletException, IOException {
DynaValidatorForm df = (DynaValidatorForm) form;
HashMap hm = (HashMap) df.get("people");
Person p = new Person();
p = new Person();
p.setDateOfBirth("07/13/1942");
p.setLastName("Ford");
p.setFirstName("Harrison");
p.setGender("M");
hm.put("1", p);
p = new Person();
p.setDateOfBirth("10/21/1956");
p.setLastName("Fisher");
p.setFirstName("Carrie");
p.setGender("F");
hm.put("2", p);
p = new Person();
p.setDateOfBirth("09/25/1951");
p.setLastName("Hamill");
p.setFirstName("Mark");
p.setGender("M");
hm.put("3", p);
return mapping.findForward("success");
}
}
基本上,這段代碼與前面的代碼相同,除了我們將Person 對(duì)象存儲(chǔ)到HashMap 中,而不是排列中之外。我們也不需要?jiǎng)?chuàng)建HashMap,因?yàn)樗梢宰鳛楸砀癯跏蓟囊徊糠謥?lái)動(dòng)態(tài)實(shí)現(xiàn)。
在JSP本身中相應(yīng)的技巧部分為:
<!--
Copyright 2004, James M Turner.
All Rights Reserved
-->
<%@ taglib uri="/WEB-INF/struts-html.tld" prefix="html" %>
<%@ taglib uri="/WEB-INF/struts-html-el.tld" prefix="html-el" %>
<%@ taglib uri="/WEB-INF/struts-tiles.tld" prefix="tiles" %>
<%@ taglib uri="/WEB-INF/c.tld" prefix="c" %>
<%@ taglib prefix="fmt" uri="/WEB-INF/fmt.tld" %>
<head>
<title>Star Wars Actor Fact Page</title>
</head>
<H1><center>Start Wars Actor Fact Page</title>
<html:form action="/processHashActorComments" >
<table border="1" width="80%">
<tr><th>Last Name</th><th>First Name</th>
<th>Date of Birth</th><th>Comment</th></tr>
<c:forEach var="people" items="${dynamicHashmapForm.map.people}">
<tr><td><c:out value="${people.value.lastName}"/></td>
<td><c:out value="${people.value.firstName}"/></td>
<td><c:out value="${people.value.dateOfBirth}"/></td>
<td><html-el:text property="comments(${people.value.lastName},
${people.value.firstName})" /></td>
</tr>
</c:forEach>
</table>
<P/>
<html:submit value="Update Comments"/>
</html:form>
記住:在初始化時(shí)填充的HashMap 值,只要表格顯示就會(huì)消失,因?yàn)楸砀袷钦?qǐng)求范圍的,而不是會(huì)話期范圍的。特別是對(duì)于我們來(lái)說(shuō)這就意味著所有的Person 對(duì)象都會(huì)消失。所以,如果我們粘貼文本域到Person bean 的注釋屬性上,在提交表格時(shí)我們將得到一個(gè)空的指針異常,因?yàn)镻erson 對(duì)象不再位于HashMap 內(nèi)(實(shí)際上,我們得到的是一個(gè)全新的空的HashMap.)。所以,我們需要將注釋存儲(chǔ)在一個(gè)單獨(dú)的并行HashMap 內(nèi),它將注釋當(dāng)作簡(jiǎn)單的字符串來(lái)存儲(chǔ)。
在上述的代碼中還須注意幾件事情。首先,因?yàn)楝F(xiàn)在正迭代HashMap條,來(lái)自c:forEach 標(biāo)記的值實(shí)際上是用于堆棧條的占位符,同時(shí)具有兩個(gè)屬性。key 屬性的值用來(lái)訪問(wèn)堆棧(在我們的例子中如字符"1", "2", "3"等等),value 屬性的值存儲(chǔ)在關(guān)鍵字之下。所以,在這種情況下,我們必須使用value 屬性來(lái)得到Person bean 的實(shí)屬性。
而且,我們需要構(gòu)造一個(gè)用于文本框的有效的Struts屬性域。在html-el 標(biāo)記庫(kù)中使用JSTL 擴(kuò)展就可以實(shí)現(xiàn)。在這種情況下,我們通過(guò)一個(gè)由演員的最后一個(gè)名字、逗號(hào)和第一個(gè)名字組成的字符串來(lái)存儲(chǔ)注釋。
最后,我們需要一個(gè)新的動(dòng)作來(lái)處理結(jié)果:
package demo;
/**
* Copyright 2004, James M. Turner.
* All Rights Reserved
*
* A Struts action that sends the new comments to the console
*/
import java.io.IOException;
import java.util.HashMap;
import java.util.Iterator;
import javax.servlet.ServletException;
import javax.servlet.http.*;
import org.apache.struts.action.*;
import org.apache.struts.validator.DynaValidatorForm;
public class ProcessHashFormAction extends Action {
public ActionForward execute(ActionMapping mapping,
ActionForm form, HttpServletRequest request,
HttpServletResponse response)
throws ServletException, IOException {
DynaValidatorForm df = (DynaValidatorForm) form;
HashMap hm = (HashMap) df.get("comments");
Iterator it = hm.keySet().iterator();
while (it.hasNext()) {
String key = (String) it.next();
String comment = (String) hm.get(key);
System.out.println(key + ":" + comment);
}
return mapping.findForward("success");
}
}
同樣,這里最大的差別是數(shù)據(jù)都是作為HashMaps 來(lái)存儲(chǔ)的。代碼獲取關(guān)鍵字(lastname,firstname),然后顯示關(guān)鍵字和在控制臺(tái)注釋: Fisher,Carrie:Leia Ford,Harrison:Han Hamill,Mark:Luke 請(qǐng)注意,當(dāng)控制返回到JSP頁(yè)時(shí),打印一個(gè)空白表格。這是因?yàn)槲覀冊(cè)诔跏蓟僮髦袆?chuàng)建的HashMap 已經(jīng)沒(méi)有了,在處理結(jié)果時(shí)我們不能重新創(chuàng)建它。你可以將該數(shù)據(jù)保存在會(huì)話期變量中,但是接著你要返回到你使用第一個(gè)方案的地方。最好是選擇一個(gè)關(guān)鍵字,在表格提交時(shí)它可以允許你在后臺(tái)對(duì)象上獲得,并且能夠總是重新創(chuàng)建需要的任何其他的表格數(shù)據(jù)。 哪種方式更好?基于排列的方案允許你將所有的數(shù)據(jù)都保存在一個(gè)bean 內(nèi),而基于堆棧的方法避免了任何會(huì)話期范圍的數(shù)據(jù)。你覺(jué)得哪種方案最好就采用哪種。 注意:包含運(yùn)行這些例子所需的所有代碼和庫(kù)的WAR 文件在http://www.blackbear.com/struts.war.上可以找到。
posted on 2005-10-24 22:10
Sung 閱讀(222)
評(píng)論(0) 編輯 收藏 所屬分類:
Struts