在Spring 4和Hibernate 4中,我可以使用Reflection通过以下代码从当前环境中获取Hibernate Configuration对象:
@Autowired LocalContainerEntityManagerFactoryBean lcemfb; EntityManagerFactoryImpl emf = (EntityManagerFactoryImpl) lcemfb.getNativeEntityManagerFactory(); SessionFactoryImpl sf = emf.getSessionFactory(); SessionFactoryServiceRegistryImpl serviceRegistry = (SessionFactoryServiceRegistryImpl) sf.getServiceRegistry(); Configuration cfg = null; try { Field field = SessionFactoryServiceRegistryImpl.class.getDeclaredField("configuration"); field.setAccessible(true); cfg = (Configuration) field.get(serviceRegistry); } catch (NoSuchFieldException | SecurityException | IllegalArgumentException | IllegalAccessException e) { e.printStackTrace(); } SchemaUpdate update = new SchemaUpdate(serviceRegistry, cfg);
使用Hibernate 5时,我必须使用some MetadataImplementor,这些对象似乎都不可用。我还试图用MetadataSources同serviceRegistry。但是它确实说这是错误的一种ServiceRegistry。
MetadataImplementor
MetadataSources
serviceRegistry
ServiceRegistry
还有其他方法可以使它正常工作吗?
我想加总Aviad的答案,以按照OP的要求完成它。
内部:
为了获得MetadataImplementor的实例,解决方法是通过Java的ServiceLoader工具注册SessionFactoryBuilderFactory的实例。然后,在引导hibernate模式时,MetadataImplementor会使用其自身的实例调用此注册服务的getSessionFactoryBuilder方法。代码参考如下:
因此,最终要获得MetadataImplementor的实例,必须实现SessionFactoryBuilderFactory并注册,以便ServiceLoader可以识别此服务:
SessionFactoryBuilderFactory的实现:
public class MetadataProvider implements SessionFactoryBuilderFactory { private static MetadataImplementor metadata; @Override public SessionFactoryBuilder getSessionFactoryBuilder(MetadataImplementor metadata, SessionFactoryBuilderImplementor defaultBuilder) { this.metadata = metadata; return defaultBuilder; //Just return the one provided in the argument itself. All we care about is the metadata :) } public static MetadataImplementor getMetadata() { return metadata; } }
为了注册以上内容,请在以下路径中创建简单的文本文件(假设它是一个maven项目,最终我们需要在类路径中提供“ META-INF”文件夹):
src/main/resources/META-INF/services/org.hibernate.boot.spi.SessionFactoryBuilderFactory
文本文件的内容应为一行(如果需要注册多个实例,甚至可以为多行),说明SessionFactoryBuilderFactory实现的完全限定的类路径。例如,对于上述类,如果您的程序包名称为“ com.yourcompany.prj”,则文件内容应为以下内容。
com.yourcompany.prj.MetadataProvider
就是这样,如果您运行应用程序,spring应用程序或独立的hibernate模式,则在启动hibernate模式后,将通过静态方法获得一个MetadataImplementor实例。
更新1:
无法通过Spring注入它。我研究了Hibernate的源代码,并且元数据对象没有存储在SessionFactory中的任何位置(这是我们从Spring获得的)。因此,不可能注入它。但是,如果您希望使用Spring的方式,则有两种选择:
LocalSessionFactoryBean-> MetadataSources-> MetadataBuilder
LocalSessionFactoryBean是您在Spring中配置的,并且具有MetadataSources对象。MetadataSources创建MetadataBuilder,后者又创建MetadataImplementor。以上所有操作均不存储任何内容,它们只是动态创建对象并返回。如果要使用MetaData实例,则应扩展和修改上述类,以便它们在返回之前存储相应对象的本地副本。这样,您可以引用MetadataImplementor。但是除非确实需要,否则我不会真正推荐这样做,因为API可能会随着时间而变化。
EntityManagerFactoryImpl emf=(EntityManagerFactoryImpl)lcemfb.getNativeEntityManagerFactory(); SessionFactoryImpl sf=emf.getSessionFactory(); StandardServiceRegistry serviceRegistry = sf.getSessionFactoryOptions().getServiceRegistry(); MetadataSources metadataSources = new MetadataSources(new BootstrapServiceRegistryBuilder().build()); Metadata metadata = metadataSources.buildMetadata(serviceRegistry); SchemaUpdate update=new SchemaUpdate(serviceRegistry,metadata); //To create SchemaUpdate // You can either create SchemaExport from the above details, or you can get the existing one as follows: try { Field field = SessionFactoryImpl.class.getDeclaredField("schemaExport"); field.setAccessible(true); SchemaExport schemaExport = (SchemaExport) field.get(serviceRegistry); } catch (NoSuchFieldException | SecurityException | IllegalArgumentException | IllegalAccessException e) { e.printStackTrace(); }