org.hibernate.MappingException: Unknown entity: annotations.Users

前端 未结 19 1491
情歌与酒
情歌与酒 2020-12-01 07:11

Consider the hierarchy :

\"enter

And the following classes and xml :

相关标签:
19条回答
  • 2020-12-01 08:15

    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:

    • programmatically re-add the mapped class using configuration.addAnnotatedClass(...)
    • create a hbm.xml file and resource mapping in hibernate_test.cfg.xml that were redundant to the existing annotations
    • scan the package where the (already mapped) class exists using external dependencies not mentioned in the original question

    However, 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!

    0 讨论(0)
提交回复
热议问题