Spring 表单:提交大量对象时出现 BindException

Spring Form: BindException when submiting large list of objects

我有这个表格,它显示了数据表中的几个对象列表。通过此表单,管理员用户可以定义特定用户对这些对象的权限。

例如,这是产品权限数据表 jsp 代码:

<sf:form action="${action}" class="form-horizontal" method="post" modelAttribute="permisosList" enctype="multipart/form-data" >
...
<table class="table table-striped table-bordered table-hover" id="tabla_permisos6">
    <thead>
        <tr>
            <th><%=Languages.getString("Producto")%></th>
            <th><%=Languages.getString("Permiso")%></th>
        </tr>
    </thead>
    <tbody>
        <c:forEach items="${permisosList.permisosProducto}" var="permiso" varStatus="counter">
            <tr>
                <sf:input path="permisosProducto[${counter.index}].producto.idProd" type="hidden"/>
                <td style="vertical-align:middle;"><c:out value="${permiso.producto.nombre}"/></td>
                <td align="center">
                    <sf:select class="form-control" path="permisosProducto[${counter.index}].permiso" type="number">
                        <sf:option value="0">
                            <c:out value="No visible" />
                        </sf:option>
                        <sf:option value="1">
                            <c:out value="Visible" />
                        </sf:option>
                        <sf:option value="2">
                            <c:out value="Descarga" />
                        </sf:option>
                    </sf:select>
                </td>
            </tr>
        </c:forEach>
    </tbody>
</table>
...

有些列表中有成百上千个对象,所以当我提交表格时,我得到spring BindException:

org.springframework.validation.BindException: org.springframework.validation.BeanPropertyBindingResult: 4 errors

Field error in object 'permisosList' on field 'permisosProducto[137].producto.idProd': rejected value []; codes [typeMismatch.permisosList.permisosProducto[137].producto.idProd,typeMismatch.permisosList.permisosProducto.producto.idProd,typeMismatch.permisosProducto[137].producto.idProd,typeMismatch.permisosProducto.producto.idProd,typeMismatch.idProd,typeMismatch.int,typeMismatch]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [permisosList.permisosProducto[137].producto.idProd,permisosProducto[137].producto.idProd]; arguments []; default message [permisosProducto[137].producto.idProd]]; default message [Failed to convert property value of type 'java.lang.String' to required type 'int' for property 'permisosProducto[137].producto.idProd'; nested exception is java.lang.NumberFormatException: For input string: ""]

Field error in object 'permisosList' on field 'permisosProducto[338].permiso': rejected value []; codes [typeMismatch.permisosList.permisosProducto[338].permiso,typeMismatch.permisosList.permisosProducto.permiso,typeMismatch.permisosProducto[338].permiso,typeMismatch.permisosProducto.permiso,typeMismatch.permiso,typeMismatch.int,typeMismatch]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [permisosList.permisosProducto[338].permiso,permisosProducto[338].permiso]; arguments []; default message [permisosProducto[338].permiso]]; default message [Failed to convert property value of type 'java.lang.String' to required type 'int' for property 'permisosProducto[338].permiso'; nested exception is java.lang.NumberFormatException: For input string: ""]

Field error in object 'permisosList' on field 'permisosProducto[573].permiso': rejected value []; codes [typeMismatch.permisosList.permisosProducto[573].permiso,typeMismatch.permisosList.permisosProducto.permiso,typeMismatch.permisosProducto[573].permiso,typeMismatch.permisosProducto.permiso,typeMismatch.permiso,typeMismatch.int,typeMismatch]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [permisosList.permisosProducto[573].permiso,permisosProducto[573].permiso]; arguments []; default message [permisosProducto[573].permiso]]; default message [Failed to convert property value of type 'java.lang.String' to required type 'int' for property 'permisosProducto[573].permiso'; nested exception is java.lang.NumberFormatException: For input string: ""]

Field error in object 'permisosList' on field 'permisosProducto[808].permiso': rejected value []; codes [typeMismatch.permisosList.permisosProducto[808].permiso,typeMismatch.permisosList.permisosProducto.permiso,typeMismatch.permisosProducto[808].permiso,typeMismatch.permisosProducto.permiso,typeMismatch.permiso,typeMismatch.int,typeMismatch]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [permisosList.permisosProducto[808].permiso,permisosProducto[808].permiso]; arguments []; default message [permisosProducto[808].permiso]]; default message [Failed to convert property value of type 'java.lang.String' to required type 'int' for property 'permisosProducto[808].permiso'; nested exception is java.lang.NumberFormatException: For input string: ""]

我已经检查过那些索引处的信息没有问题,所以没有任何空字段(那为什么一旦提交数据就会有空字段?)。我还注意到这些索引并不总是相同的。也许如果我重新启动 tomcat 或者我在几分钟后重试,错误中的索引是不同的。 并且只发生在包含至少 200 个以上对象的列表中。并且在到达控制器之前抛出异常。

这是我的 PermisoList class:

public class PermisoList {
    List<PermisoProductoForm> permisosProducto;
    List<PermisoCarpetaForm> permisosCarpeta;
    List<PermisoTipoDocForm> permisosTipoDoc;
    List<PermisoAgenteClienteForm> permisosAgenteCliente;
    List<PermisoIdiomaForm> permisosIdioma;
    List<PermisoPaisForm> permisosPais;
    List<PermisoEmpresaForm> permisosEmpresa;
    List<PermisoUsuarioProductoForm> permisosUsuarioProducto;   
...
}

我的 PermisoProductoForm class:

public class PermisoProductoForm {

    private Producto producto;
    private int permiso;
...
}

还有我的 Producto class:

@Entity
@Table(name = "Producto")
public class Producto {

    @Id
    @GeneratedValue
    private int idProd;

    private String codigo;
    private String nombre;
    private String marca;
    private boolean activo;

    @ManyToOne
    @JoinColumn(name = "idFamilia")
    private Familia familia;

    @OneToMany(mappedBy = "producto", fetch = FetchType.LAZY, cascade = CascadeType.ALL)
    private List<Documento> documentacion;

    @ManyToOne
    @JoinColumn(name = "idEmpresa")
    private Empresa empresa;

    @OneToMany(mappedBy = "primaryKey.producto", fetch = FetchType.LAZY, cascade = CascadeType.ALL)
    private List<PermisoUsuarioProducto> permisosProducto;

    @OneToMany(mappedBy = "productoRelacionado", fetch = FetchType.LAZY)
    private List<TemaForo> temasForo;
...
}

此外,我已经尝试了一些我发现的解决方案,例如将此代码添加到控制器中:

@InitBinder
public void initBinder(WebDataBinder binder) {
    binder.setAutoGrowCollectionLimit(2048);
}

或在 jsp 代码中的表单标签内添加 enctype="multipart/form-data"

有谁知道为什么会这样或者知道如何解决这个问题?我现在 100% 卡住了。


编辑: 我已经完成@angcap 回答的内容(将 int 更改为 Integer 类型并在 initBinder 方法中添加 CustomEditor),现在不再抛出此 BindException。所以到达控制器。

现在发生的情况是那些具有空字符串的索引现在具有 'null' 值(由于 CustomEditor)。正如我所说,我在提交表单之前检查了信息,我可以看到列表中的所有对象都没有问题,没有一个元素带有空字段。

那么为什么 Spring 在列表很大时对数据执行此操作?它似乎对随机索引执行此操作...


编辑 2: 这是一个 TOMCAT 问题! 我已经在 Pivotal tc Server (v3. 1) 和 binder.setAutoGrowCollectionLimit(2048); 解决方案一起工作,没有任何异常或空字符串或 null。功能齐全。

所以问题出在tomcat (v8.0) 配置。有人知道这件事吗?


编辑 3: 我接受@angcap 的回答,因为它确实解决了我最初的问题。按照他说的去做,就避免了 BindException。

您似乎为数字字段 idProdpermiso 发布了空字符串。检查发布的数据或尝试注册 CustomNumberEditor 允许在您的控制器中使用 @initBinder 的 empyValues。