Отображение перечисления в таблицу с аннотацией спящего режима
У меня есть таблица DEAL и таблица DEAL_TYPE. Я бы хотел отобразить этот код:
public class Deal {
DealType type;
}
public enum DealType {
BASE("Base"), EXTRA("Extra");
}
Проблема в том, что данные уже существуют в базе данных. И мне трудно сопоставить классы с базой данных.
База данных выглядит примерно так:
TABLE DEAL {
Long id;
Long typeId;
}
TABLE DEAL_TYPE {
Long id;
String text;
}
Я знаю, что могу использовать простые отношения @OneToMany от сделки до типа сделки, но я бы предпочел использовать перечисление. Возможно ли это?
Я почти заработал, используя тип EnumType.ORDINAL. Но, к сожалению, мои идентификаторы в моей таблице типов транзакций не являются последовательными и не начинаются с 1.
Любые предложения?
Ответы
Ответ 1
Спящий режим ужасен у Эннума. Это странный провал в противном случае довольно хороший ORM. "Самый простой" способ обойти это - объявить свой Enum пользовательским типом спящего режима. К счастью, Hibernate написал пример реализации, который вы можете вложить в ваше приложение:
http://www.hibernate.org/265.html
Они даже содержат инструкции по его использованию. Это шаблон, который я использую всякий раз, когда мне в конечном итоге нужно сохранить переписку.
Ответ 2
Я создал аналогичный класс, подобный тому, который предлагается только для спящего режима, только настраиваемый, и нет необходимости создавать новый тип только для этой настойчивости.
Может использоваться как
@Type(type = "ro.raisercostin.hibernate.EnumUserType", parameters = @Parameter(name = "type", value = "DealType"))
DealType dealType;
Я добавил реализацию ParameterizedType для поддержки переданного параметра.
public class EnumUserType implements UserType, ParameterizedType {
private static final int[] SQL_TYPES = { Types.VARCHAR };
private Class clazz = null;
public EnumUserType() {
}
@Override
public void setParameterValues(Properties parameters) {
String className = (String) parameters.get("type");
try {
this.clazz = Class.forName(className);
} catch (ClassNotFoundException e) {
throw new RuntimeException("Couldn't get the class for name [" + className + "].", e);
}
}
public int[] sqlTypes() {
return SQL_TYPES;
}
public Class returnedClass() {
return clazz;
}
public Object nullSafeGet(ResultSet resultSet, String[] names, Object owner) throws HibernateException,
SQLException {
String name = resultSet.getString(names[0]);
Object result = null;
if (!resultSet.wasNull()) {
result = Enum.valueOf(clazz, name);
}
return result;
}
public void nullSafeSet(PreparedStatement preparedStatement, Object value, int index) throws HibernateException,
SQLException {
if (null == value) {
preparedStatement.setNull(index, Types.VARCHAR);
} else {
preparedStatement.setString(index, ((Enum) value).name());
}
}
public Object deepCopy(Object value) throws HibernateException {
return value;
}
public boolean isMutable() {
return false;
}
public Object assemble(Serializable cached, Object owner) throws HibernateException {
return cached;
}
public Serializable disassemble(Object value) throws HibernateException {
return (Serializable) value;
}
public Object replace(Object original, Object target, Object owner) throws HibernateException {
return original;
}
public int hashCode(Object x) throws HibernateException {
return x.hashCode();
}
public boolean equals(Object x, Object y) throws HibernateException {
if (x == y) {
return true;
}
if ((null == x) || (null == y)) {
return false;
}
return x.equals(y);
}
}
Ответ 3
Вы могли бы аннотировать enum с помощью @Entity
и использовать кумулятор custoumn для создания экземпляров перечисления с помощью Enum.valueOf
Объявление enum выглядит следующим образом:
@Entity
@Table(name = "node_interface_type")
@Tuplizer(impl = EnumTuplizer.class)
public enum Type {
WIRED, WIRELESS, WIRELESS_SENSOR_NODE;
@Id
public String name = toString();
}
И Tuplizer:
public class EnumTuplizer extends PojoEntityTuplizer {
public EnumTuplizer(EntityMetamodel entityMetamodel, PersistentClass mappedEntity) {
super(entityMetamodel, mappedEntity);
}
@Override
protected Instantiator buildInstantiator(final PersistentClass persistentClass) {
return new Instantiator() {
@Override
public Object instantiate(Serializable id) {
try {
return Enum.valueOf(
(Class) persistentClass.getClass().getClassLoader().loadClass(persistentClass.getClassName()),
(String) id
);
} catch (ClassNotFoundException e) {
throw new AssertionError(e);
}
}
@Override
public Object instantiate() {
throw new UnsupportedOperationException();
}
@Override
public boolean isInstance(Object object) {
throw new UnsupportedOperationException();
}
};
}
}
Ответ 4
Если вы хотите использовать объект только для чтения, вы можете использовать @Formula
и @Enumerated
. Попробуйте что-то вроде:
@Entity
public class Deal {
@Formula("(select text from DEAL_TYPE dt where dt.id = typeId)")
@Enumerated(EnumType.STRING)
DealType type;
}
Ответ 5
Хотя это далеко не идеально, моим решением этой проблемы было использование EnumStringType и денормализованное обновляемое представление.