我有一个p:selectOneMenu的问题,无论我做什么我都无法让JSF调用JPA实体上的setter。 JSF验证失败,显示以下消息:
form:location:验证错误:值无效
我有几个相同类型的其他类(即连接表类),但在我的生活中不能让这个工作。
如果有人可以针对此类问题提出一些故障排除/调试技巧,我们将不胜感激。
使用日志语句我已经验证了以下内容:
Conveter
返回正确的非null
值。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]
答案 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");
}
因此,基于上述逻辑,此问题在逻辑上至少可能具有以下原因:
equals()
方法丢失或损坏。Converter
,则它会在getAsObject()
中返回错误的对象。也许它甚至是null
。要解决它:
@ViewScoped
而不是@RequestScoped
应该修复它。还要确保不在<f:selectItem(s)>
的getter方法中执行业务逻辑,而是在@PostConstruct
或动作事件(侦听器)方法中执行业务逻辑。如果您依赖于特定的请求参数,那么您需要将它们显式地存储在@ViewScoped
bean中,或者在后续请求中重新传递它们,例如: <f:param>
。另请参阅How to choose the right bean scope? equals()
方法。这已经在标准Java类型上完成,例如java.lang.String
,java.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。Converter
的操作并相应地进行修复。有关指南,另请参阅Conversion Error setting value for 'null Converter'如果您使用java.util.Date
作为<f:convertDateTime>
的可用项目,请确保您不会忘记模式中的全部时间部分。另请参阅"Validation Error: Value is not valid" error from f:datetimeConverter。selectOneMenu
wiki page 如果有人可以针对此类问题提出一些故障排除/调试技巧,我们将不胜感激。
在这里问一个明确而具体的问题。不要问太广泛的问题;)
答案 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;
}