83

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

表单:位置:验证错误:值无效

我在其他几个相同类型的类(即连接表类)上进行了此工作,但我一生都无法让这个工作。

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

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

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

3 回答 3

147

验证失败并显示消息“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. 如果Converter涉及自定义,则它返回了错误的对象getAsObject()。也许它是偶数null

要解决它:

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

也可以看看:


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

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

于 2012-01-30T19:35:50.590 回答
2

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

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

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

于 2015-02-12T15:18:15.897 回答
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”。
于 2018-08-16T06:00:35.200 回答