验证错误:值无效


80

我对ap:selectOneMenu有问题,无论我做什么,我都无法让JSF调用JPA实体上的setter。JSF验证失败,并显示以下消息:

形式:位置:验证错误:值无效

我正在同一类型的其他几个类上工作(例如,联接表类),但是我一生都无法工作。

如果有人可以针对此类问题提出一些故障排除/调试提示,将不胜感激。

使用日志语句,我已经验证了以下内容:

  1. Conveter是回到正确的,非null数值。
  2. 我的JPA实体中没有Bean验证。
  3. 该设置器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] 

Answers:


143

验证失败,并显示以下消息:“ form:location:验证错误:值无效”

该错误归结为所选的项目<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");
}

因此,根据上述逻辑,此问题从逻辑上可以至少具有以下原因:

  1. 可用项目列表中缺少所选项目。
  2. equals()表示所选项目的类的方法丢失或损坏。
  3. 如果Converter涉及到某个自定义,则它在中返回了错误的对象getAsObject()。也许甚至null

解决方法:

  1. 确保在后续请求期间保留完全相同的列表,尤其是在多个级联菜单的情况下。在大多数情况下,制作Bean@ViewScoped而不是@RequestScoped应该修复它。还要确保您不要在的getter方法中执行业务逻辑<f:selectItem(s)>,而在@PostConstructaction event(listener)方法中执行业务逻辑。如果您依赖于特定的请求参数,则需要将它们显式存储在@ViewScopedBean中,或通过例如在后续请求中重新传递它们<f:param>。另请参见如何选择正确的bean作用域?
  2. 确保equals()正确实施该方法。这已经是做对了标准的Java类型,如java.lang.Stringjava.lang.Number等,但不一定对自定义对象/豆/ entites的。另请参见实施平等合同的正确方法。如果您已经在使用String,请确保正确配置了请求字符编码。如果它包含特殊字符,并且JSF配置为将输出呈现为UTF-8,但将输入解释为例如ISO-8859-1,则它将失败。另请参见通过PrimeFaces输入组件检索的Unicode输入已损坏
  3. 调试/记录自定义操作,Converter并进行相应的修复。有关准则,另请参见“空转换器”的“转换错误”设置值。如果您将java.util.Date用作“可用的项目” <f:convertDateTime>,请确保不要忘记模式中的全时部分。另请参见f:datetimeConverter中的“验证错误:值无效”错误

也可以看看:


如果有人可以针对此类问题提出一些故障排除/调试提示,将不胜感激。

只是在这里问一个明确而具体的问题。不要问太多问题;)


@BalusC:该equals检查在mojarra代码中的确切位置。我的情况有点复杂。我创建了自己的自定义组件,该组件允许用户进行复杂的无线电布局。如果我只有一个无线电组(在我的自定义组件下面的f:selectItems),它会很好地工作。但是,随着布局变得更加复杂(多个无线电组,每个无线电组都有自己的f:selectItems,但是都共享相同的选择),我必须在ui:repeat中包含f:selectItems,然后ui:repeat才在我的自定义组件下。然后,我遇到了这个问题。我希望看到处理这个钻嘴鱼科码
升范

另外我的f:selectItems是String类型,所以我确定这不是转换问题,我认为这是原因,因为在验证阶段不存在f:selectItems列表。
唐潘

您能否详细说明如何使用f:param将状态发送到托管Bean,以便它可以重新创建“初始状态” ????谢谢。
奥古斯特·桑切斯

1
有类似的错误。我正在使用SelectItemsConverter,并且此组件位于禁用的字段集中-因此也被禁用。这是已知行为吗?
Gilberto 2014年

如果您使用enum来显示项目,并且您的后备Bean具有用于在Bean中设置值的String类型,而不使用任何转换器,则您将得到相同的错误。那就是我最后的结果。
菲利普·普德维拉

2

就我而言,我忘记了实现正确的获取/设置方法。之所以发生,是因为我在开发过程中更改了许多属性。

如果没有适当的get方法,JSF将无法恢复您选择的项目,并且发生BalusC在其答案的项目1中所说的:

1。可用项目列表中缺少所选项目。如果可用项目列表由请求范围的Bean提供服务,而该请求范围的Bean在后续请求中未正确重新初始化,或者在getter方法中错误地执行了业务工作,导致它以某种方式返回了另一个列表,则可能会发生这种情况。


1

这可能是转换器问题,也可能是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”。

就像在#2中被接受并被高度评价的答案中所指出的那样
库克特耶
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.