«1. Проблема

В этой статье будет обсуждаться проблема org.hibernate.MappingException: неизвестный объект и решения как для Hibernate, так и для среды Spring и Hibernate.

2. Отсутствует или недействительна аннотация @Entity

Наиболее распространенной причиной исключения сопоставления является просто класс сущности, в котором отсутствует аннотация @Entity:

public class Foo implements Serializable {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;

    public Foo() {
        super();
    }

    public long getId() {
        return id;
    }
    public void setId(long id) {
        this.id = id;
    }
}

Другая возможность заключается в том, что он может иметь неправильный тип Аннотация @Entity:

import org.hibernate.annotations.Entity;

@Entity
public class Foo implements Serializable {
    ...

Устаревший org.hibernate.annotations.Entity — это неправильный тип сущности для использования — нам нужен javax.persistence.Entity:

import javax.persistence.Entity;

@Entity
public class Foo implements Serializable {
    ...

3. MappingException В Spring

Конфигурация Hibernate в Spring включает загрузку SessionFactory из сканирования аннотаций через LocalSessionFactoryBean:

@Bean
public LocalSessionFactoryBean sessionFactory() {
    LocalSessionFactoryBean sessionFactory = new LocalSessionFactoryBean();
    sessionFactory.setDataSource(restDataSource());
    ...
    return sessionFactory;
}

В этой простой конфигурации Session Factory Bean отсутствует ключевой ингредиент, и тест пытается использовать SessionFactory завершится ошибкой:

...
@Autowired
private SessionFactory sessionFactory;

@Test(expected = MappingException.class)
@Transactional
public void givenEntityIsPersisted_thenException() {
    sessionFactory.getCurrentSession().saveOrUpdate(new Foo());
}

Исключение, как и ожидалось, — MappingException: Unknown entity:

org.hibernate.MappingException: Unknown entity: 
com.baeldung.ex.mappingexception.persistence.model.Foo
    at o.h.i.SessionFactoryImpl.getEntityPersister(SessionFactoryImpl.java:1141)

Теперь есть два решения этой проблемы — два способа сообщить LocalSessionFactoryBean о классе сущности Foo.

Мы можем указать, в каких пакетах искать классы сущностей в пути к классам:

sessionFactory.setPackagesToScan(
  new String[] { "com.baeldung.ex.mappingexception.persistence.model" });

Или мы можем просто зарегистрировать классы сущностей непосредственно в фабрике сеансов:

sessionFactory.setAnnotatedClasses(new Class[] { Foo.class });

С любой из этих дополнительных строк конфигурации , тест теперь будет работать правильно и будет пройден.

4. MappingException With Hibernate

Теперь посмотрим на ошибку при использовании только Hibernate:

public class Cause4MappingExceptionIntegrationTest {

    @Test
    public void givenEntityIsPersisted_thenException() throws IOException {
        SessionFactory sessionFactory = configureSessionFactory();

        Session session = sessionFactory.openSession();
        session.beginTransaction();
        session.saveOrUpdate(new Foo());
        session.getTransaction().commit();
    }

    private SessionFactory configureSessionFactory() throws IOException {
        Configuration configuration = new Configuration();
        InputStream inputStream = this.getClass().getClassLoader().
          getResourceAsStream("hibernate-mysql.properties");
        Properties hibernateProperties = new Properties();
        hibernateProperties.load(inputStream);
        configuration.setProperties(hibernateProperties);

        // configuration.addAnnotatedClass(Foo.class);

        ServiceRegistry serviceRegistry = new ServiceRegistryBuilder().
          applySettings(configuration.getProperties()).buildServiceRegistry();
        SessionFactory sessionFactory = configuration.buildSessionFactory(serviceRegistry);
        return sessionFactory;
    }
}

Файл hibernate-mysql.properties содержит свойства конфигурации Hibernate:

hibernate.connection.username=tutorialuser
hibernate.connection.password=tutorialmy5ql
hibernate.connection.driver_class=com.mysql.jdbc.Driver
hibernate.dialect=org.hibernate.dialect.MySQL5Dialect
hibernate.connection.url=jdbc:mysql://localhost:3306/spring_hibernate4_exceptions
hibernate.show_sql=false
hibernate.hbm2ddl.auto=create

Выполнение этого теста приведет к приводит к тому же исключению сопоставления:

org.hibernate.MappingException: 
  Unknown entity: com.baeldung.ex.mappingexception.persistence.model.Foo
    at o.h.i.SessionFactoryImpl.getEntityPersister(SessionFactoryImpl.java:1141)

Как, вероятно, уже ясно из приведенного выше примера, в конфигурации отсутствует добавление метаданных класса сущности — Foo — в конфигурацию: ~ ~~

configuration.addAnnotatedClass(Foo.class);

Это исправляет тест, который теперь может сохранять сущность Foo.

5. Заключение

В этой статье показано, почему может возникнуть исключение при отображении неизвестной сущности и как решить проблему, когда она возникает, сначала на уровне сущности, затем с помощью Spring и Hibernate и, наконец, только с помощью Hibernate.

Реализацию всех примеров исключений можно найти в проекте github — это проект на основе Eclipse, поэтому его легко импортировать и запускать как есть.