Отображение перечисления в таблицу с аннотацией гибернации - PullRequest
28 голосов
/ 10 апреля 2009

У меня есть таблица 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 от сделки к типу сделки, но я бы предпочел использовать enum. Возможно ли это?

Я почти все заработал, используя тип EnumType.ORDINAL. Но, к сожалению, мои идентификаторы в таблице типов сделок не являются последовательными и не начинаются с 1.

Есть предложения?

Ответы [ 5 ]

17 голосов
/ 10 апреля 2009

Hibernate отчасти ужасен в Enums. Это странный недостаток ORM. Самый простой способ обойти это - объявить свой Enum настраиваемым спящим типом. К счастью, Hibernate написал пример реализации, которую вы можете дословно зашифровать в свое приложение:

http://www.hibernate.org/265.html

Они даже включают инструкции о том, как его использовать. Это шаблон, который я использую всякий раз, когда у меня возникает необходимость сохранять перечисления.

14 голосов
/ 22 апреля 2010

Я создал похожий класс, подобный классу, предложенному только hibernate, который можно настраивать, и нет необходимости создавать новый тип только для этой персистенции.

Может использоваться как

@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);
    }
}
7 голосов
/ 04 апреля 2011

Вы можете аннотировать перечисление с помощью @Entity и использовать cupoumn tuplizer для создания экземпляров перечисления с 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();
}

А туплизер это:

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();
            }
        };
    }
}
3 голосов
/ 15 августа 2012

Если вы хотите использовать объект только для чтения, вы можете использовать @Formula и @Enumerated. Попробуйте что-то вроде:

@Entity
public class Deal {
   @Formula("(select text from DEAL_TYPE dt where dt.id = typeId)")
   @Enumerated(EnumType.STRING)
   DealType type;
}
1 голос
/ 10 апреля 2009

Хотя это было далеко от идеала, мое решение этой проблемы заключалось в использовании EnumStringType и денормализованного обновляемого представления.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...