验证错误:值无效

Validation Error: Value is not valid

提问人:klonq 提问时间:1/31/2012 最后编辑:BalusCklonq 更新时间:8/18/2018 访问量:92191

问:

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

form:location: 验证错误: 值无效

我有这个在相同类型的其他几个类(即,联接表类)上工作,但在我的一生中无法让这个工作。

如果有人可以为此类问题提供一些故障排除/调试技巧,我们将不胜感激。

使用 log 语句,我验证了以下内容:

  1. 返回正确的非值。Conveternull
  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] 
验证 JSF JPA 转换器 SelectOneMenu

评论


答:

149赞 BalusC 1/31/2012 #1

验证失败,并显示消息“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");
}

所以,根据上面的逻辑,这个问题在逻辑上至少可以有以下几个原因:

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

要解决它:

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

另请参阅:


如果有人可以为此类问题提供一些故障排除/调试技巧,我们将不胜感激。

在这里问一个明确而具体的问题。不要问太宽泛的问题;)

评论

0赞 Thang Pham 11/17/2012
@BalusC:在mojarra代码中,这种检查究竟在哪里发生。我的情况有点复杂。我创建了自己的自定义组件,允许用户进行复杂的无线电布局。如果我只有一个单选组(f:selectItems 在我的自定义组件下方),它可以正常工作。但是,随着布局变得越来越复杂(多个单选组,每个单选组都有自己的 f:selectItems,但都共享相同的选择),我必须在 ui:repeat 中拥有 f:selectItems,然后 ui:repeat 在我的自定义组件下。然后我遇到了这个问题。我想看看处理这个问题的 mojarra 代码equals
0赞 Thang Pham 11/19/2012
此外,我的 f:selectItems 是 String 类型,所以我确定这不是转换问题,我认为这是原因,因为在验证阶段 f:selectItems 列表不存在。
0赞 Agustí Sánchez 8/20/2014
您能否详细说明如何使用 f:param 将状态发送到托管 Bean,以便它可以重新创建“初始状态”???谢谢。
1赞 Gilberto 9/4/2014
有类似的错误。我正在使用 SelectItemsConverter,并且此组件位于禁用的字段集中 - 因此它也被禁用。这是已知的行为吗?
0赞 Philip Puthenvila 9/24/2015
如果您使用枚举来显示项目,并且您的后备 Bean 具有用于设置 Bean 中的值的 String 类型,并且不使用任何转换器,则将收到相同的错误。这就是我来到这里的原因。
2赞 Henrique Liberato 2/12/2015 #2

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

如果没有正确的 get 方法,JSF 就无法恢复您选择的项目,并且会发生 BalusC 在他的答案的第 1 项中所说的:

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

1赞 Rizla Shareefdeen 8/16/2018 #3

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

评论

0赞 Kukeltje 8/16/2018
就像 #2 中接受和高度赞成的答案中所说的那样