【问题标题】:Validation Error: Value is not valid验证错误:值无效
【发布时间】:2015-11-03 22:14:05
【问题描述】:

我遇到了 p:selectOneMenu 的问题,无论我做什么,我都无法让 JSF 调用 JPA 实体上的设置器。 JSF 验证失败并显示此消息:

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

我在其他几个相同类型的类(即连接表类)上也有这个工作,但我一辈子都不能让这个工作。

如果有人能为此类问题提供一些故障排除/调试提示,将不胜感激。

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

  1. Conveter 返回正确的非 null 值。
  2. 我的 JPA 实体中没有 Bean 验证。
  3. setter 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] 

【问题讨论】:

    标签: validation jsf jpa converter selectonemenu


    【解决方案1】:

    验证失败并显示消息“form:location: Validation Error: Value is not valid”

    此错误归结为在处理表单提交请求期间,所选项目与任何嵌套 &lt;f:selectItem(s)&gt; 标记指定的任何可用选择项目值都不匹配。

    作为防止篡改/黑客请求的一部分,JSF 将重申所有可用的选择项值并测试selectedItem.equals(availableItem) 是否针对至少一个可用项值返回true。如果没有一个项目值匹配,那么您将得到这个验证错误。

    这个过程基本上如下所示,bean.getAvailableItems() 虚构地代表&lt;f:selectItem(s)&gt; 定义的可用选择项的整个列表:

    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 应该可以修复它。还要确保您不在&lt;f:selectItem(s)&gt; 的getter 方法中执行业务逻辑,而是在@PostConstruct 或动作事件(侦听器)方法中执行业务逻辑。如果您依赖特定的请求参数,那么您需要将它们显式存储在 @ViewScoped bean 中,或者通过例如在后续请求中重新传递它们。 &lt;f:param&gt;。另见How to choose the right bean scope?
    2. 确保正确实施equals() 方法。这已经在标准 Java 类型(例如 java.lang.Stringjava.lang.Number 等)上正确完成,但不一定在自定义对象/beans/实体上完成。另见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 用作&lt;f:convertDateTime&gt; 的可用项目,请确保您不要忘记模式中的全职部分。另见"Validation Error: Value is not valid" error from f:datetimeConverter

    另见:


    如果有人能为这类问题提供一些故障排除/调试提示,将不胜感激。

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

    【讨论】:

    • @BalusC:在 mojarra 代码中的确切位置,equals 检查发生。我的情况有点复杂。我创建了自己的自定义组件,允许用户拥有复杂的收音机布局。如果我只有一个无线电组(f:selectItems 就在我的自定义组件下方),它工作得很好。然而,随着布局变得更加复杂(多个单选组,每个组都有自己的 f:selectItems 但都共享相同的选择),我必须在 ui:repeat 中有 f:selectItems,然后 ui:repeat 在我的自定义组件下。然后我遇到了这个问题。我想看看处理这个的 mojarra 代码
    • 我的 f:selectItems 也是 String 类型,所以我确定这不是转换问题,我认为这是因为在验证阶段 f:selectItems 列表不存在。
    • 您能否详细说明如何使用 f:param 将状态发送到托管 bean,以便它可以重新创建“初始状态”?谢谢。
    • 有类似的错误。我正在使用SelectItemsConverter,这个组件在一个禁用的字段集中——所以它也被禁用了。这是一种已知的行为吗?
    • 如果您使用枚举来显示要显示的项目,并且您的支持 bean 具有用于设置 bean 中的值的字符串类型并且不使用任何转换器,您将得到相同的错误。我就是这样来到这里的。
    【解决方案2】:

    就我而言,我忘记实现正确的 get/set 方法。这是因为我在开发过程中改变了很多属性。

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

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

    【讨论】:

      【解决方案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”。

      【讨论】:

      • 就像在接受和高度支持的答案中的 #2 中所述
      猜你喜欢
      相关资源
      最近更新 更多