Spring Web MVC中映射数组及Collection类

在开发过程中,遇到需要在页面中映射一个对象,而这个对象中有一个List的属性,于是来CSND问,结果不要说答案,回复的都少得可怜。
于是不得不自己找搜索,中文的,没有发现答案,后来在Spring的官方网站发现答案,特此发文以做纪念。
要映射的对象:

class Question{
   private String questionString;
   private int questionId;
   private  List<Option> options = new ArrayList<Option>();
}
class Option{
   private int optionId;
   private String optionString;
}

 jsp代码

<form:form commandName="surveyQuestion">
  <form:hidden path="questionId"/>
  Question: <form:input path="questionTitle"/>
  <c:forEach items="${surveyQuestion.options}" var="option" varStatus="counter">
    <spring:bind path="surveyQuestion.options[${counter.index}].optionString">
 Option<input type="text" name="<%= status.getExpression() %>" value="<%= status.getValue() %>"<br>
    </spring:bind>
  </c:forEach>
</form:form>

 大致上找到的答案都是这样说,这样的写法也的确可以保证页面正确显示form及数据,但是当点击提交按钮时却出现


org.springframework.beans.InvalidPropertyException: Invalid property 'options[0]' of bean class [Question]: Index of out of bounds in property path 'options[0]'; nested exception is java.lang.IndexOutOfBoundsException: Index: 0, Size: 0

没有办法只好再次去查找,终于找到,有人在论坛(非CSDN论坛)说道,需要在Question的默认的constructor中加入以下语句:
options = ListUtils.lazyList(new ArrayList<Option>(),FactoryUtils.instantiateFactory(Option.class));
应该是Spring在初始化的时候存在一些问题。没有时间看源代码。再次提交,成功了!


后来发现,还是存在一点儿问题,就是当对一个已存在的Question添加Option时仍会出现同样的问题,解决方案请继续看。

在上文中我提到了在command对象的构造器中添加:options = ListUtils.lazyList(new ArrayList<Option>(),FactoryUtils.instantiateFactory(Option.class));来解决org.springframework.beans.InvalidPropertyException: Invalid property 'options[0]' of bean class [Question]: Index of out of bounds in property path 'options[0]'; nested exception is java.lang.IndexOutOfBoundsException: Index: 0, Size: 0的问题,但是在开发过程中发现,仍存在着问题。

使用上面的方法,当这个Question对象是第一次录入的时候就没有问题,更改Option也没有问题,但是如果用户增加Question中的option的数目时,还会遇到上面的问题,当然,这个时候不是index:0, size: 0,而是index是question中线有的options的数目。

从网上找了很久,没有答案,迫不得已去看Spring的源代码,发现Spring在绑定对象的时候,先从Cache中拿出原来的对象,并且根据现在提供的数据,一次拿出原数据,显而易见问题就出在这里,因为原来Question中只有5个option,现在用户添加了一个,那么当Spring去找第6个Option时,肯定会出现我们上面遇到的问题。因为仅仅读了关于绑定这一部分代码,所以我不想更改Spring。

那么我们重新把注意力放在构造器的更改上面,既然通过增加一行代码可以让新的Question对象正确地去的数据,那么完全可以把这一行代码添加到getOptions方法中,这样一来,在每次取得options这个List时,都对他进行的Lazy处理。

准备改的时候,想到,添加这一行代码仅仅是为了Spring的应用,那么如果当我们把这一部分换成Struts或者其他框架的时候,这一行代码显然是多余的,那么反而不如在Spring的controller里面进行处理。

修改Controller,工作了。

Question question = ....//getQuestion;
        if(question==null){
            Question = new Question();
            question.setOptions(ListUtils.lazyList(new ArrayList<Option>(),FactoryUtils.instantiateFactory(Option.class)));
        }else{
            question.setOptions(ListUtils.lazyList(question.getOptions(),FactoryUtils.instantiateFactory(Option.class)));
        }

可以看到,在else里面使用的是question中已经拿到的option,而不是新建一个ArrayList对象。