我有一个的Hibernate映射Java对象, JKL
,这是充满了一堆正常休眠可映射领域(如字符串和整数)的。
我增加了一个新的嵌入式域将其(其生活在同一个表-不是一个映射), asdf
,这是一个fj.data.Option<ASDF>
我做了这一个选项,使之清楚,这个领域实际上可能并不包含任何(而不是必须处理null
我每次访问)。
如何设置在我映射JKL.hbm.xml
文件? 我想Hibernate来自动转换一个null
数据库到none
的fj.data.Option<ASDF>
当获取对象。 还应的一个非空实例转换ASDF
到some
的fj.data.Option<ASDF>
是否有其他弄虚作假,我有什么关系? 谢谢。
我建议引入FunctionalJava的Option
中的访问者(getter和setter),同时使Hibernate来处理这允许是一个简单的java场null
。
例如,对于一个可选Integer
字段:
// SQL
CREATE TABLE `JKL` (
`JKL_ID` INTEGER PRIMARY KEY,
`MY_FIELD` INTEGER DEFAULT NULL
)
您可以直接映射Hibernate的私有字段:
// Java
@Column(nullable = true)
private Integer myField;
然后,您可以引入Option
在访问边界:
// Java
public fj.data.Option<Integer> getMyField() {
return fj.data.Option.fromNull(myField);
}
public void setMyField(fj.data.Option<Integer> value) {
myField = value.toNull();
}
请问您的需求的工作?
您可以使用Hibernate的自定义映射类型。 文档是在这里 。 下面是一个类似的实例映射Scala的选项到Hibernate映射。
简单地说,你将需要延长org.hibernate.UserType
接口。 你也可以创建一个通用类型的基类与JKL
-typed分型,类似于你在Scala的例子中看到。
我认为使用getter/setter
是简单的,但这里是我所做的,使其工作的例子:
(它工作正常的数字和字符串,而不是日期(误差@Temporal
注释))。
import com.cestpasdur.helpers.PredicateHelper;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Optional;
import org.apache.commons.lang.ObjectUtils;
import org.apache.commons.lang.StringUtils;
import org.hibernate.HibernateException;
import org.hibernate.usertype.UserType;
import org.joda.time.DateTime;
import java.io.Serializable;
import java.sql.*;
public class OptionUserType implements UserType {
@Override
public int[] sqlTypes() {
return new int[]{
Types.NULL
};
}
@Override
public Class returnedClass() {
return Optional.class;
}
@Override
public boolean equals(Object o, Object o2) throws HibernateException {
return ObjectUtils.equals(o, o2);
}
@Override
public int hashCode(Object o) throws HibernateException {
assert (o != null);
return o.hashCode();
}
@Override
public Optional<? extends Object> nullSafeGet(ResultSet rs, String[] names, Object owner) throws HibernateException, SQLException {
return Optional.fromNullable(rs.getObject(names[0]));
}
@VisibleForTesting
void handleDate(PreparedStatement st, Date value, int index) throws SQLException {
st.setDate(index, value);
}
@VisibleForTesting
void handleNumber(PreparedStatement st, String stringValue, int index) throws SQLException {
Double doubleValue = Double.valueOf(stringValue);
st.setDouble(index, doubleValue);
}
@Override
public void nullSafeSet(PreparedStatement st, Object value, int index) throws SQLException {
if (value != null) {
if (value instanceof Optional) {
Optional optionalValue = (Optional) value;
if (optionalValue.isPresent()) {
String stringValue = String.valueOf(optionalValue.get());
if (StringUtils.isNotBlank(stringValue)) {
if (PredicateHelper.IS_DATE_PREDICATE.apply(stringValue)) {
handleDate(st, new Date(DateTime.parse(stringValue).getMillis()), index);
} else if (StringUtils.isNumeric(stringValue)) {
handleNumber(st, stringValue, index);
} else {
st.setString(index, optionalValue.get().toString());
}
} else {
st.setString(index, null);
}
} else {
System.out.println("else Some");
}
} else {
//TODO replace with Preconditions guava
throw new IllegalArgumentException(value + " is not implemented");
}
} else {
st.setString(index, null);
}
}
@Override
public Object deepCopy(Object o) throws HibernateException {
return o;
}
@Override
public boolean isMutable() {
return false;
}
@Override
public Serializable disassemble(Object o) throws HibernateException {
return (Serializable) o;
}
@Override
public Object assemble(Serializable serializable, Object o) throws HibernateException {
return serializable;
}
@Override
public Object replace(Object original, Object target, Object owner) throws HibernateException {
return original;
}
}