Consider the hierarchy :
And the following classes and xml :
I did not find the accepted answer helpful in resolving the exception encountered in my code. And while not technically incorrect, I was also not satisfied with others' suggestions to introduce redundancy:
configuration.addAnnotatedClass(...)
hbm.xml
file and resource mapping in hibernate_test.cfg.xml
that were redundant to the existing annotationsHowever, I found 2 possible solutions that I wanted to share, both of which independently resolved the exception I encountered in my own code.
I was having the same MappingException
as @ron (using a very nearly identical HibernateUtil
class):
public final class HibernateUtil {
private static SessionFactory sessionFactory = null;
private static ServiceRegistry serviceRegistry = null;
private HibernateUtil() {}
public static synchronized SessionFactory getSessionFactory() {
if ( sessionFactory == null ) {
Configuration configuration = new Configuration().configure("hibernate_test.cfg.xml");
serviceRegistry = new StandardServiceRegistryBuilder()
.applySettings(configuration.getProperties())
.build();
sessionFactory = configuration.buildSessionFactory( serviceRegistry );
}
return sessionFactory;
}
// exception handling and closeSessionFactory() omitted for brevity
}
Within my hibernate_test.cfg.xml
configuration file, I have the required class mapping:
<mapping class="myPackage.Device"/>
And my Device
class is properly annotated with the javax.persistence.Entity
annotation:
package myPackage.core;
import javax.persistence.*;
@Entity
@Table( name = "devices" )
public class Device {
//body omitted for brevity
}
Two Possible Solutions:
First, I am using Hibernate 5.2, and for those using Hibernate 5 this solution using a Metadata
object to build a SessionFactory
should work. It also appears to be the currently recommended native bootstrap mechanism in the Hibernate Getting Started Guide :
public static synchronized SessionFactory getSessionFactory() {
if ( sessionFactory == null ) {
// exception handling omitted for brevity
serviceRegistry = new StandardServiceRegistryBuilder()
.configure("hibernate_test.cfg.xml")
.build();
sessionFactory = new MetadataSources( serviceRegistry )
.buildMetadata()
.buildSessionFactory();
}
return sessionFactory;
}
Second, while Configuration
is semi-deprecated in Hibernate 5, @ron didn't say which version of Hibernate he was using, so this solution could also be of value to some.
I found a very subtle change in the order of operations when instantiating and configuring Configuration
and ServiceRegistry
objects to make all the difference in my own code.
Original order (Configuration
created and configured prior to ServiceRegistry
):
public static synchronized SessionFactory getSessionFactory() {
if ( sessionFactory == null ) {
// exception handling omitted for brevity
Configuration configuration = new Configuration().configure("hibernate_test.cfg.xml");
serviceRegistry = new StandardServiceRegistryBuilder()
.applySettings( configuration.getProperties() )
.build();
sessionFactory = configuration.buildSessionFactory( serviceRegistry );
}
return sessionFactory;
}
New order (ServiceRegistry
created and configured prior to Configuration
):
public static synchronized SessionFactory getSessionFactory() {
if ( sessionFactory == null ) {
// exception handling omitted for brevity
serviceRegistry = new StandardServiceRegistryBuilder()
.configure("hibernate_test.cfg.xml")
.build();
sessionFactory = new Configuration().buildSessionFactory( serviceRegistry );
}
return sessionFactory;
}
At the risk of TLDR, I will also point out that with respect to hibernate_test.cfg.xml
my testing suggests that the configuration.getProperties()
method only returns <property />
elements, and <mapping />
elements are excluded. This appears consistent with the specific use of the terms 'property' and 'mapping' within the API documentation for Configuration
. I will concede that this behaviour may have something to do with the failure of applySettings()
to yield the mapping data to the StandardServiceRegistryBuilder
. However, this mapping data should already have been parsed during configuration of the Configuration
object and available to it when buildSessionFactory()
is called. Therefore, I suspect this may be due to implementation-specific details regarding resource precedence when a ServiceRegistry
is passed to a Configuration
object's buildSessionFactory()
method.
I know this question is several years old now, but I hope this answer saves somebody the hours of research I spent in deriving it. Cheers!