提问人:klonq 提问时间:1/31/2012 最后编辑:BalusCklonq 更新时间:8/18/2018 访问量:92191
验证错误:值无效
Validation Error: Value is not valid
问:
我对 p:selectOneMenu 有问题,无论我做什么,我都无法让 JSF 调用 JPA 实体上的 setter。JSF 验证失败,并显示以下消息:
form:location: 验证错误: 值无效
我有这个在相同类型的其他几个类(即,联接表类)上工作,但在我的一生中无法让这个工作。
如果有人可以为此类问题提供一些故障排除/调试技巧,我们将不胜感激。
使用 log 语句,我验证了以下内容:
- 返回正确的非值。
Conveter
null
- 我的 JPA 实体中没有 Bean 验证。
- 从不调用二传手。
setLocation(Location location)
这是我能做的最简单的例子,它根本行不通:
<h:body>
<h:form id="form">
<p:messages id="messages" autoUpdate="true" />
<p:selectOneMenu id="location" value="#{locationStockList.selected.location}" converter="locationConverter">
<p:ajax event="change" update=":form:lblLocation"/>
<f:selectItems value="#{locationStockList.locationSelection}"/>
</p:selectOneMenu>
</h:form>
</h:body>
转炉:
@FacesConverter(forClass=Location.class, value="locationConverter")
public class LocationConverter implements Converter, Serializable {
private static final Logger logger = Logger.getLogger(LocationConverter.class.getName());
@Override
public Object getAsObject(FacesContext context, UIComponent component, String value) {
if (value.isEmpty())
return null;
try {
Long id = Long.parseLong(value);
Location location = ((LocationManagedBean) context.getApplication().getELResolver().getValue(context.getELContext(), null, "location")).find(id);
logger.log(Level.SEVERE, "Converted {0} to {1}" , new Object[] {value, location});
return location;
} catch (NumberFormatException e) {
return new Location();
}
}
@Override
public String getAsString(FacesContext context, UIComponent component, Object value) {
if (value == null || value.toString().isEmpty() || !(value instanceof Location))
return "";
return String.valueOf(((Location) value).getId());
}
}
控制台输出:
// Getter method
INFO: Current value=ejb.locations.Location[id=null, name=null, latitude=0.0, longitude=0.0]
// Session Bean
INFO: Finding ejb.locations.Location with id=3
// Session Bean
INFO: ### Returning : ejb.locations.Location[id=3, name=mdmd, latitude=4.5, longitude=2.3]
// Converter
SEVERE: Converted 3 to ejb.locations.Location[id=3, name=mdmd, latitude=4.5, longitude=2.3]
// Getter method -> Where did my selected Location go ??
INFO: Current value=ejb.locations.Location[id=null, name=null, latitude=0.0, longitude=0.0]
答:
验证失败,并显示消息“form:location: Validation Error: Value is not valid”
此错误归结为所选项目与在处理表单提交请求期间由任何嵌套标记指定的任何可用选择项目值不匹配。<f:selectItem(s)>
作为防止篡改/黑客攻击请求的一部分,JSF 将重申所有可用的选择项值,并测试是否返回至少一个可用项值。如果没有一个项目值匹配,则你将收到此验证错误。selectedItem.equals(availableItem)
true
这个过程基本上如下,其中虚构地表示可用选择项目的整个列表,定义如下:bean.getAvailableItems()
<f:selectItem(s)>
String submittedValue = request.getParameter(component.getClientId());
Converter converter = component.getConverter();
Object selectedItem = (converter != null) ? converter.getAsObject(context, component, submittedValue) : submittedValue;
boolean valid = false;
for (Object availableItem : bean.getAvailableItems()) {
if (selectedItem.equals(availableItem)) {
valid = true;
break;
}
}
if (!valid) {
throw new ValidatorException("Validation Error: Value is not valid");
}
所以,根据上面的逻辑,这个问题在逻辑上至少可以有以下几个原因:
- 可用项目列表中缺少所选项目。
- 表示所选项的类的方法丢失或损坏。
equals()
- 如果涉及自定义,则它返回了 中的错误对象。也许它甚至.
Converter
getAsObject()
null
要解决它:
- 确保在后续请求期间保留完全相同的列表,尤其是在多个级联菜单的情况下。在大多数情况下,制作豆子而不是应该修复它。此外,请确保不要在 的 getter 方法中执行业务逻辑,而是在 或 操作事件(侦听器)方法中执行业务逻辑。如果你依赖于特定的请求参数,那么你需要将它们显式地存储在 bean 中,或者通过例如 .另请参阅 如何选择合适的 Bean 范围?
@ViewScoped
@RequestScoped
<f:selectItem(s)>
@PostConstruct
@ViewScoped
<f:param>
- 确保正确实现该方法。这在标准 Java 类型(如 、 等)上已经正确完成,但不一定在自定义对象/beans/entites 上。另请参阅实现平等合同的正确方法。如果您已经在使用 ,请确保正确配置了请求字符编码。如果它包含特殊字符,并且 JSF 配置为将输出呈现为 UTF-8,但将输入解释为 .ISO-8859-1,那么它将失败。另请参阅 a.o. 通过 PrimeFaces 输入组件检索的 Unicode 输入已损坏。
equals()
java.lang.String
java.lang.Number
String
- 调试/记录自定义的操作并相应地修复它。有关指南,另请参阅“null 转换器”的转换错误设置值 如果您将 用作可用项目,请确保不要忘记模式中的全时部分。另请参阅 f:datetimeConverter 中的“验证错误:值无效”错误。
Converter
java.util.Date
<f:convertDateTime>
另请参阅:
如果有人可以为此类问题提供一些故障排除/调试技巧,我们将不胜感激。
在这里问一个明确而具体的问题。不要问太宽泛的问题;)
评论
equals
就我而言,我忘记实现正确的获取/设置方法。之所以发生这种情况,是因为我在开发过程中更改了很多属性。
如果没有正确的 get 方法,JSF 就无法恢复您选择的项目,并且会发生 BalusC 在他的答案的第 1 项中所说的:
1 .可用项目列表中缺少所选项目。如果可用项列表由请求范围的 Bean 提供,而该 Bean 在后续请求中未正确重新初始化,或者在 getter 方法中错误地执行业务工作,导致它以某种方式返回不同的列表,则可能会发生这种情况。
这可能是转换器问题,也可能是 DTO 问题。 尝试通过在对象 DTO 中添加 hashCode() 和 equals() 方法来解决这个问题;在上面的场景中,您可以在 Location 对象类中生成这些方法,这些方法在此处指示为“DTO”。
例:
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + (int) (id ^ (id >>> 32));
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
Location other = (Location) obj;
if (id != other.id)
return false;
return true;
}
- 请注意,上面的示例适用于类型为“long”的“id”。
评论