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