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。
您似乎为数字字段 idProd
和 permiso
发布了空字符串。检查发布的数据或尝试注册 CustomNumberEditor 允许在您的控制器中使用 @initBinder 的 empyValues。
我有这个表格,它显示了数据表中的几个对象列表。通过此表单,管理员用户可以定义特定用户对这些对象的权限。
例如,这是产品权限数据表 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。
您似乎为数字字段 idProd
和 permiso
发布了空字符串。检查发布的数据或尝试注册 CustomNumberEditor 允许在您的控制器中使用 @initBinder 的 empyValues。