学习Hibernate时,经常会遇到一些小问题,这里将介OSGi平台的插件类加载机制使得Hibernate无法正确加载分布在不同插件内部的模型对象与O/R映射文件问题的解决方法。
OpenCore是在OSGi规范上构建的微内核(Microkenerl),基于纯组件(Pure Plugin)开放源码企业应用软件平台。
OpenCore数据层实现OSGi上集成Hibernate,Hibernate及其依赖库作为一个单独的插件,这样带来一个问题,就是OSGi平台的插件类加载机制使得Hibernate无法正确加载分布在不同插件内部的模型对象与O/R映射文件。本文讨论解决方案:
模型对象(Domain Objects)插件
模型对象(Domain Objects)集中到独立的插件(Bundle)内,Hibernate插件依赖这些模型对象插件。这是最简单的,也是比较糟糕的方式,比较小的基于OSGi的项目可以这也作做。
Eclipse-BuddyPolicy与Eclipse-RegisterBuddy方式
Equinox(Eclipse提供的OSGi实现)平台特有的方式,允许插件(Bundle)声明自己的伙伴,让“伙伴插件”来动态加载本插件的类,这也是Hiberate与Equinox集成的官方解决方案。这种方式模型对象无需部署在单独的插件内,与业务插件部署在一起即可,Hibernate插件也无须依赖模型对象。
具体做法如下:
首先,Hibernate插件(名称,例如org.opengoss.orm.hibernate)声明自身可以作为伙伴插件,自描述文件(MANIFEST.MF) 加入描述:Eclipse-BuddyPolicy: registered
然后,模型对象的业务插件中把Hibernate插件加入为伙伴,自描述文件(MANIFEST.MF) 加入描述:Eclipse-RegisterBuddy:org.opengoss.orm.hibernate
Eclipse Extension Point方式
这是我们目前实现的方式,通过标准的Eclipse扩展点与扩展机制,我们在Hibernate插件中plugin.xml配置文件中声明下述扩展点,在模型对象插件中声明扩展,例如:
Hibernate插件的启动中,用代码配置生成SessionFactory,代码如下:
- public void start(BundleContext context) throws Exception {
- Configuration configuration = new Configuration().configure
(new File ("./etc/org.opengoss.database.hibernate/hibernate.cfg.xml"));- Class[] domainClasses = getDomainClasses();
- for (Class domainClass : domainClasses) {
- configuration.addClass(domainClass);
- }
- sessionFactory = configuration.buildSessionFactory();
- Dictionarynew Hashtable props.put("scope", "APPLICATION");
- props.put("uid", "Hibernate:SessionFactory");
- registration = context.registerService
(SessionFactory.class.getName(), sessionFactory, props);- }
- private Class[] getDomainClasses() throws Exception {
- List domainClasses = new ArrayList();
- IExtensionPoint point = registry.getExtensionPoint
(IConstants.DOMAIN_OBJECT_EXTENSION_POINT);- IExtension[] extensions = point.getExtensions();
- for (IExtension extension : extensions) {
- IConfigurationElement[] elements = extension.getConfigurationElements();
- for (IConfigurationElement configurationElement : elements) {
- Bundle bundle = pluginContext.getBundleBySymbolId
(extension.getNamespaceIdentifier());- Class domainClass = bundle.loadClass
(configurationElement.getAttribute("class"));- domainClasses.add(domainClass);
- }
- }
- return domainClasses.toArray(new Class[domainClasses.size()]);
- }
注意:Hibernate内部的类加载机制实在无法令人满意,尽管我们在这种方式中已经加载所有的模型类对象,但Hibernate内部仍然会调用Class.forName()去试图加载。
【编辑推荐】