在Succeeding with Struts的前面安裝部分,我間接提到了DynaForms在運行期內(nèi)可以動態(tài)的控制表格大小。換句話說,就是能夠根據(jù)需要得到5行、或者10行、或者15行長的表格。可能有點不明智,我把這種策略的實際實現(xiàn)作為一種練習留給了讀者自己。在接下來的幾個月內(nèi),我收到了幾十個讀者的請求,他們請求給出詳細的實現(xiàn)細節(jié),所以這個月我將用兩種不同的方法來實現(xiàn)動態(tài)調(diào)整的表格。
第一個方法就是我在前面的欄目中提到的那個方法,將尺寸參數(shù)留給DynaForm 的form-property 屬性來實現(xiàn)。為了演示詳細過程,我們來看看一個非常簡單的應(yīng)用:添加關(guān)于不同Star Wars 演員的注釋。在這個應(yīng)用中我們感興趣的關(guān)鍵事實是:演員的數(shù)量在表格配置中動態(tài)設(shè)定,而不是在struts-config.xml文件中動態(tài)設(shè)定。
首先,我們先來看看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> |
如你所見,這是一個相當簡單的配置文件,只定義了一個表格和兩個動作。第一個動作,/setupForm,用來在初始顯示之前配置表格;另一個動作,/processActorComments 用來處理用戶輸入的注釋。
在這個文件中有兩個重要的事情需要注意,它們對于事態(tài)的發(fā)展很關(guān)鍵:
1. people 表格屬性定義為demo.Person[] 類型(即demo.Person的一個排列),但不給出任何size 參數(shù)。這就為要創(chuàng)建的排列產(chǎn)生了一個占位符,但是沒有任何例示的實排列。
2. 這兩個動作將表格定義在會話期范圍內(nèi)。這是很關(guān)鍵的,因為用戶在填寫數(shù)值之后提交表格時,數(shù)值在動作執(zhí)行之前已經(jīng)填充到表格內(nèi)了。這就意味著沒有機會手動創(chuàng)建具有恰當空位數(shù)的排列,正如你在表格顯示之前在SetupFormAction 類中看到的情況一樣。換句話說,當表格提交時,必須已經(jīng)有恰當?shù)目瘴粊斫邮鼙砀裰担ㄒ荒鼙WC這個的方法就是在會話期范圍內(nèi)就已經(jīng)有了這個表格。
基本上在Person bean 中是沒有值的,他只是一個具有l(wèi)astName、 firstName、 dateOfBirth、gender 和comment字段的普通bean。源文件包括在WAR 文件內(nèi)。
現(xiàn)在我們來看看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");
}
} |
這一次也沒有許多東西要看的。execute 方法要做的第一件事情,和任何基于DynaForm的動作所做的一樣,就是將泛型ActionForm 類放到DynaValidatorForm內(nèi)。這就使得我們可以在表格上使用get和set 方法。第二件事情就是,創(chuàng)建一個具有三個元素的類型Person 的排列。在這個方法中,尺寸是硬布線的,在實際應(yīng)用中可以從數(shù)據(jù)庫中選擇一個尺寸。我們需要考慮的重要事情是排列應(yīng)該在代碼中創(chuàng)建,而不是由Struts引擎自己創(chuàng)建。這樣行數(shù)可根據(jù)應(yīng)用要求由代碼隨意指定。
一旦排列已經(jīng)確定,方法將創(chuàng)建三個Person 類實例并賦與數(shù)值。同樣,在實際的應(yīng)用中可通過一個循環(huán)來實現(xiàn),這個循環(huán)不斷地從數(shù)據(jù)庫中讀取行和填充表格行。最后,動作返回成功,導致Struts轉(zhuǎn)移控制到displayForm.jsp 頁。
<!--
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> |
同樣,這里也沒有很多東西要看的,他與我們上一篇文章查看固定長度的行時的代碼完全一樣。該頁迭代行(記住在JSTL中我們必須使用map 屬性來獲得到DynaForm 屬性的訪問),顯示演員的姓、名和出生日期,并提供文本域以便輸入注釋。
當我們聚焦我們的瀏覽器合請求時,http://localhost:8080/struts/setupForm.do (假設(shè)你把struts.war 文件放在你本地機器的Tomcat 內(nèi)),將會出現(xiàn)下列頁面:
Start Wars Actor Fact Page
一旦表格提供,另一個簡單的Struts動作來處理結(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");
}
} |
在實際的應(yīng)用中,這就是數(shù)據(jù)寫回到數(shù)據(jù)庫的地方。在這種情況下,他只將數(shù)據(jù)倒在控制臺上所以我們可以看到他是正確收到的。假設(shè)我們?yōu)槊總€演員都填充了恰當?shù)闹担覀冊诳刂婆_上會看到下列內(nèi)容:
Harrison Ford:Indiana Jones
Carrie Fisher:Postcards from the Edge
Mark Hamill:Wing Commander |
正如我在文章開頭提到的一樣,還有另一個方法可以解決這個問題,而且它不需要使用會話期范圍內(nèi)的表格。這個方法就是使用HashMaps 來存儲行。我們來看看使用HashMaps編寫的同一段代碼:
首先,我們添加一個新表格到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 來存儲每個人的數(shù)據(jù)。另外,我們需要一個新的HashMap 來存儲注釋,原因我稍后再解釋。我們也需要一個新的動作來填充數(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 對象存儲到HashMap 中,而不是排列中之外。我們也不需要創(chuàng)建HashMap,因為它可以作為表格初始化的一部分來動態(tài)實現(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> |
記住:在初始化時填充的HashMap 值,只要表格顯示就會消失,因為表格是請求范圍的,而不是會話期范圍的。特別是對于我們來說這就意味著所有的Person 對象都會消失。所以,如果我們粘貼文本域到Person bean 的注釋屬性上,在提交表格時我們將得到一個空的指針異常,因為Person 對象不再位于HashMap 內(nèi)(實際上,我們得到的是一個全新的空的HashMap.)。所以,我們需要將注釋存儲在一個單獨的并行HashMap 內(nèi),它將注釋當作簡單的字符串來存儲。
在上述的代碼中還須注意幾件事情。首先,因為現(xiàn)在正迭代HashMap條,來自c:forEach 標記的值實際上是用于堆棧條的占位符,同時具有兩個屬性。key 屬性的值用來訪問堆棧(在我們的例子中如字符"1", "2", "3"等等),value 屬性的值存儲在關(guān)鍵字之下。所以,在這種情況下,我們必須使用value 屬性來得到Person bean 的實屬性。
而且,我們需要構(gòu)造一個用于文本框的有效的Struts屬性域。在html-el 標記庫中使用JSTL 擴展就可以實現(xiàn)。在這種情況下,我們通過一個由演員的最后一個名字、逗號和第一個名字組成的字符串來存儲注釋。
最后,我們需要一個新的動作來處理結(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 來存儲的。代碼獲取關(guān)鍵字(lastname,firstname),然后顯示關(guān)鍵字和在控制臺注釋:
Fisher,Carrie:Leia
Ford,Harrison:Han
Hamill,Mark:Luke
請注意,當控制返回到JSP頁時,打印一個空白表格。這是因為我們在初始化操作中創(chuàng)建的HashMap 已經(jīng)沒有了,在處理結(jié)果時我們不能重新創(chuàng)建它。你可以將該數(shù)據(jù)保存在會話期變量中,但是接著你要返回到你使用第一個方案的地方。最好是選擇一個關(guān)鍵字,在表格提交時它可以允許你在后臺對象上獲得,并且能夠總是重新創(chuàng)建需要的任何其他的表格數(shù)據(jù)。
哪種方式更好?基于排列的方案允許你將所有的數(shù)據(jù)都保存在一個bean 內(nèi),而基于堆棧的方法避免了任何會話期范圍的數(shù)據(jù)。你覺得哪種方案最好就采用哪種。
注意:包含運行這些例子所需的所有代碼和庫的WAR 文件在http://www.blackbear.com/struts.war.上可以找到。
posted on 2005-10-24 22:10
Sung 閱讀(216)
評論(0) 編輯 收藏 所屬分類:
Struts