验证错误:值无效

时间:2012-01-30 19:13:10

标签: validation jsf jpa converter selectonemenu

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

  

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

我有几个相同类型的其他类(即连接表类),但在我的生活中不能让这个工作。

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

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

  1. Conveter返回正确的非null值。
  2. 我的JPA实体中没有Bean验证。
  3. 永远不会调用setter setLocation(Location location)
  4. 这是我能做的最简单的例子,它根本不起作用:

    <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] 
    

3 个答案:

答案 0 :(得分:128)

  

验证失败,并显示消息“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
  4. 要解决它:

    1. 确保在后续请求期间保留完全相同的列表,尤其是在多个级联菜单的情况下。在大多数情况下,使bean @ViewScoped而不是@RequestScoped应该修复它。还要确保不在<f:selectItem(s)>的getter方法中执行业务逻辑,而是在@PostConstruct或动作事件(侦听器)方法中执行业务逻辑。如果您依赖于特定的请求参数,那么您需要将它们显式地存储在@ViewScoped bean中,或者在后续请求中重新传递它们,例如: <f:param>。另请参阅How to choose the right bean scope?
    2. 确保正确实施equals()方法。这已经在标准Java类型上完成,例如java.lang.Stringjava.lang.Number等,但不一定在自定义对象/ beans / entites上。另见Right way to implement equals contract。如果您已经在使用String,请确保正确配置了请求字符编码。如果它包含特殊字符,并且JSF配置为将输出呈现为UTF-8,但将输入解释为例如ISO-8859-1,那么它将失败。另见a.o. Unicode input retrieved via PrimeFaces input components become corrupted
    3. 调试/记录自定义Converter的操作并相应地进行修复。有关指南,另请参阅Conversion Error setting value for 'null Converter'如果您使用java.util.Date作为<f:convertDateTime>的可用项目,请确保您不会忘记模式中的全部时间部分。另请参阅"Validation Error: Value is not valid" error from f:datetimeConverter
    4. 另见:


        

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

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

答案 1 :(得分:2)

在我的情况下,我忘了实现正确的get / set方法。之所以发生这种情况,是因为我在开发过程中改变了许多属性。

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

  

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

答案 2 :(得分: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'。