6

有一个对象ObjectA有一个列表ObjectBTreeMap里面有一个ObjectB。这TreeMap有一个String作为键和List另一个对象的一个ObjectC​​作为值。this TreeMapand listinside 已在jsp使用s:iteratorand时显示,s:textfield并且它正在正确显示。即 s:textfield 中的“值”是正确的。现在,修改文本字段时出现问题。我们如何在动作类的 ObjectC 中捕获修改后的值?使用此处给出的代码,键(“Key1”)出现在操作中,但值为 null。

Java 代码

public class ObjectA implements Serializable {
private Integer attr1;
private List<ObjectB> objB;
//...getters and setters....
public class ObjectB implements Serializable {
private Integer attr11;
private TreeMap<String,List<ObjectC>> allPlainFields;
// ...getters and setters....
public class ObjectC implements Serializable {
private Integer attr111;
public String attr112;
// ...getters and setters....

JSP 代码

<s:iterator value="objA.objB" var="currentObjB" status="currentGroupStatus">

  <s:iterator value="#currentObjB.allPlainFields" var="parentMap" status="headerStatus">
     <s:iterator value="#parentMap.value" var="fieldList" status="fieldStatus">

       <s:textfield  name="objA.objB[%{#currentGroupStatus.index}].allPlainFields['%{#parentMap.key}'][%{#fieldStatus.index}].attr112"/>


</s:iterator>                       

 </s:iterator> 

呈现的 HTML: <input type="text" id="review-act1_objA_objB_0__allPlainFields_'Key1'__6__attr112" value="Correct Value" name="objA.objB[0].allPlainFields['Key1'][0].attr112">

eclipse的“变量”视图中的对象结构显示

objA    Object A  (id=955)  
objB    ArrayList<E>  (id=966)  
    elementData Object[10]  (id=967)    
        [0] ObjectB  (id=968)   
            allPlainFields  TreeMap<K,V>  (id=972)  
                comparator  null    
                descendingMap   null    
                entrySet    TreeMap$EntrySet  (id=979)  
                keySet  null    
                modCount    1   
                navigableKeySet null    
                root    TreeMap$Entry<K,V>  (id=980)    
                size    1   
                values  null    

        [1] ObjectB  (id=969)   
        [2] ObjectB  (id=970)   
        [3] ObjectB  (id=971)   
        [4] null    
        [5] null    
        [6] null    
        [7] null    
        [8] null    
        [9] null    
    modCount    4   
    size    4   

****在 Eclipse“变量”视图中,allPlainFields 的值为**:** {Key1=}

编辑(2013 年 2 月 27 日)

试过这个但没有用。这些值出现在 jsp 上,但在提交时,它们不会起作用:

Action课堂上:

private TreeMap<String,ObjectCList> testTreeMap = new TreeMap<String,ObjectCList>();
//get,set and setting two keys in map "mykey1" and "mykey2"

ObjectCList课堂上:

private ArrayList<ObjectC> paramMdlList;
 //default constructor, get, set

JSP

<s:form id="test-map" method="post">
<s:iterator value="testTreeMap" var="pMap" status="hStatus">
     <li><label><s:property value="%{#pMap.key}" /></label> 
        <s:iterator value="%{#pMap.value.paramMdlList}" var="pList" status="innerStatus">
            <s:textfield name="testTreeMap['%{#pMap.key}'].paramMdlList[%{#innerStatus.index}].attr111"/>
            <s:textfield name="testTreeMap['%{#pMap.key}'].paramMdlList[%{#innerStatus.index}].attr112"/> 

        </s:iterator>


    </li>

</s:iterator>
<s:submit value=" " type='button' id="btnh1" action="saveTreeMap">
            <span>Save TreeMap</span>
</s:submit>
</s:form>

提交表单时,会调用updateTreeMap的方法。地图按此处action所述打印:

public String updateTreeMap(){

    for (Map.Entry<String, ObjectCList> entry : testTreeMap.entrySet())
    {
        System.out.println(entry.getKey() + "/" + entry.getValue());
    }

    return SUCCESS;

}

什么是“打印”:mykey1/mykey2/ 即空值

下面的屏幕显示了来自 jsp 的值 屏幕显示值来自 jsp

4

2 回答 2

6

根据您的最新更新。如果您使用的是TreeMapStruts2,则无法正确确定其中的元素类型。将声明testTreeMapfrom更改TreeMapMap

private Map<String,ObjectCList> testTreeMap = new TreeMap<String,ObjectCList>();

或者testTreeMapcom.opensymphony.xwork2.util.Element注解来告诉Struts2 map里面的元素是什么类型。

@Element(value = ObjectCList.class)
private TreeMap<String,ObjectCList> testTreeMap = new TreeMap<String,ObjectCList>();
于 2013-02-27T09:37:11.017 回答
5

我变得好奇并做了其他实验。

我发现Lists 中List的 s 和Maps 中Map的 s(以及所有插值)都没有声明为接口(List, Map),或者它们的实现(ArrayList, HashMap, TreeMap)都没有被XWork Converter.

所有的测试用例都失败了。

也许这是我的错,如果是这样的话,我们真的需要一些OGNL专家,因为在整个网络上都没有谈论这个。

然后我尝试了我非常确定会起作用的方法:以纯粹的方式将这些信息封装在自定义对象OOP中。

它奏效了:)

代替

private ArrayList<ArrayList<String>> outerObjects;

你可以在你的动作中使用

private ArrayList<OuterObject> outerObjects; 

/* GETTERS AND SETTERS */
public ArrayList<OuterObject> getOuterObjects() {
    return outerObjects;
}
public void setOuterObjects(ArrayList<OuterObject> outerObjects) {
    this.outerObjects = outerObjects;
}

外部对象定义:

/* ELSEWHERE IN YOUR PROJECT... */
public class OuterObject{
    private ArrayList<InnerObject> innerObjects;

    /* GETTERS AND SETTERS */
    public ArrayList<InnerObject> getInnerObjects() {
        return innerObjects;
    }
    public void setInnerObjects(ArrayList<InnerObject> innerObjects) {
        this.innerObjects = innerObjects;
    }       
}

内部对象定义:

public class InnerObject{
    String innerField;

    /* GETTERS AND SETTERS */       
    public String getInnerField() {         
        return innerField;
    }
    public void setInnerField(String innerField) {
        this.innerField = innerField;
    }

}

Action的可选execute()方法来测试预设值:

        InnerObject innerObj1 = new InnerObject();
        innerObj1.setInnerField("Inner Value 1");

        ArrayList<InnerObject> innerObjArrayList = new ArrayList<InnerObject>();
        innerObjArrayList.add(innerObj1);

        OuterObject outerObj1 = new OuterObject();      
        outerObj1.setInnerObjects(innerObjArrayList);

        outerObjects = new ArrayList<OuterObject>();
        outerObjects.add(outerObj1);

JSP:_

    <s:form>
        <s:textfield name="outerObjects[0].innerObjects[0].innerField" />
        <s:submit/>
    </s:form>

(迭代时,只需使用[%{#stat.index}]for Lists 和['%{#stat.index}']for Maps)

相同的解决方案适用于每一种可迭代的结构(可能除了需要调用方法的Guava东西)。.create()

当然,这并不是在每种情况下都很方便,在您的示例中,您已经拥有一个巨大的结构,这几乎会使它翻倍,但它可以工作,它是 OOP,您的 OGNL 会更清晰(因为名称),但似乎是唯一的方法。

注意:类必须是真正的独立类,而不是自动装配对象失败的Inner Classes另一种情况。OGNL

希望有帮助


编辑

那么你所需要的,只是多了一层:

改变这个:

private TreeMap<String,List<ObjectC>> allPlainFields;

对此

private TreeMap<String,ObjectX> allPlainFields;

并创建一个ObjectX包含私有字段的List<ObjectC>.

它会起作用的。

于 2013-02-21T17:59:10.933 回答