最简单的方法是让您的所有 JPA 实体从这样的基本实体扩展:
public abstract class BaseEntity<T extends Number> implements Serializable {
private static final long serialVersionUID = 1L;
public abstract T getId();
public abstract void setId(T id);
@Override
public int hashCode() {
return (getId() != null)
? (getClass().getSimpleName().hashCode() + getId().hashCode())
: super.hashCode();
}
@Override
public boolean equals(Object other) {
return (other != null && getId() != null
&& other.getClass().isAssignableFrom(getClass())
&& getClass().isAssignableFrom(other.getClass()))
? getId().equals(((BaseEntity<?>) other).getId())
: (other == this);
}
@Override
public String toString() {
return String.format("%s[id=%d]", getClass().getSimpleName(), getId());
}
}
请注意,拥有正确的equals()
(and hashCode()
) 很重要,否则您将面临Validation Error: Value is not valid。这些Class#isAssignableFrom()
测试是为了避免在例如基于 Hibernate 的代理上的测试失败,而无需退回到特定于 Hibernate 的Hibernate#getClass(Object)
辅助方法。
并拥有这样的基本服务(是的,我忽略了您正在使用 Spring 的事实;这只是为了给出基本的想法):
@Stateless
public class BaseService {
@PersistenceContext
private EntityManager em;
public BaseEntity<? extends Number> find(Class<BaseEntity<? extends Number>> type, Number id) {
return em.find(type, id);
}
}
并按如下方式实现转换器:
@ManagedBean
@ApplicationScoped
@SuppressWarnings({ "rawtypes", "unchecked" }) // We don't care about BaseEntity's actual type here.
public class BaseEntityConverter implements Converter {
@EJB
private BaseService baseService;
@Override
public String getAsString(FacesContext context, UIComponent component, Object value) {
if (value == null) {
return "";
}
if (modelValue instanceof BaseEntity) {
Number id = ((BaseEntity) modelValue).getId();
return (id != null) ? id.toString() : null;
} else {
throw new ConverterException(new FacesMessage(String.format("%s is not a valid User", modelValue)), e);
}
}
@Override
public Object getAsObject(FacesContext context, UIComponent component, String value) {
if (value == null || value.isEmpty()) {
return null;
}
try {
Class<?> type = component.getValueExpression("value").getType(context.getELContext());
return baseService.find((Class<BaseEntity<? extends Number>>) type, Long.valueOf(submittedValue));
} catch (NumberFormatException e) {
throw new ConverterException(new FacesMessage(String.format("%s is not a valid ID of BaseEntity", submittedValue)), e);
}
}
}
请注意,它被注册为 a@ManagedBean
而不是 a @FacesConverter
。这个技巧允许您通过例如在转换器中注入服务@EJB
。另请参阅如何在 @FacesConverter 中注入 @EJB、@PersistenceContext、@Inject、@Autowired 等?因此,您需要将其引用为converter="#{baseEntityConverter}"
而不是converter="baseEntityConverter"
.
如果您碰巧经常将这种转换器用于UISelectOne
/UISelectMany
组件(<h:selectOneMenu>
和朋友),您可能会发现OmniFaces SelectItemsConverter
更有用。它根据可用的值进行转换,<f:selectItems>
而不是每次都进行(可能很昂贵)数据库调用。