OpenCore: OSGi上部署Hibernate的四種方式
OpenCore是在OSGi規(guī)范上構建的微內核(Microkenerl),基于純組件(Pure Plugin)開放源碼企業(yè)應用軟件平臺。OpenCore數(shù)據(jù)層實現(xiàn)OSGi上集成Hibernate,Hibernate及其依賴庫作為一個單獨的插件,這樣帶來一個問題,就是OSGi平臺的插件類加載機制使得Hibernate無法正確加載分布在不同插件內部的模型對象與O/R映射文件。本文討論四種解決方案:
51CTO編輯推薦:OSGi入門與實踐全攻略
模型對象(Domain Objects)插件
模型對象(Domain Objects)集中到獨立的插件(Bundle)內,Hibernate插件依賴這些模型對象插件。這是最簡單的,也是比較糟糕的方式,比較小的基于OSGi的項目可以這也作做。
依賴方式:
業(yè)務插件------->Hibernate插件
| |
| \ \| /
|----------- 模型插件
/
改進的模型對象(Domain Objects)插件
把模型對象插件當作Hibernate插件的Fragments,依賴方式如圖:
業(yè)務插件------->Hibernate插件
/|\
|
模型插件
Eclipse-BuddyPolicy與Eclipse-RegisterBuddy方式
Equinox(Eclipse提供的OSGi實現(xiàn))平臺特有的方式,允許插件(Bundle)聲明自己的伙伴,讓“伙伴插件”來動態(tài)加載本插件的類,這也是Hiberate與Equinox集成的官方解決方案。這種方式模型對象無需部署在單獨的插件內,與業(yè)務插件部署在一起即可,Hibernate插件也無須依賴模型對象。
具體做法如下:
首先,Hibernate插件(名稱,例如org.opengoss.orm.hibernate)聲明自身可以作為伙伴插件,自描述文件(MANIFEST.MF) 加入描述:
Eclipse-BuddyPolicy: registered
然后,模型對象的業(yè)務插件中把Hibernate插件加入為伙伴,自描述文件(MANIFEST.MF) 加入描述:
Eclipse-RegisterBuddy:org.opengoss.orm.hibernate
具體說明文檔:
http://www.hibernate.org/311.html
http://www.ibm.com/developerworks/cn/opensource/os-ecl-osgi/index.html
注意:這種方式無法保證在Hibernate最新版本中應用成功。大家可以再試試:)
Eclipse Extension Point方式
這是我們目前實現(xiàn)的方式,通過標準的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內部的類加載機制實在無法令人滿意,盡管我們在這種方式中已經(jīng)加載所有的模型類對象,但Hibernate內部仍然會調用Class.forName()去試圖加載。所以,我們不得不在其自描述文件(MANIFEST.MF) 中加入描述:
DynamicImport-Package: *
結論:我們傾向于第四種方式,由Eclipse的擴展點功能來完成這一職責。不贊成第三種在OSGi規(guī)范層作改進的方式,OSGi本身的類加載機制設計非常優(yōu)美,Buddy插件破壞了這種優(yōu)美。
您正在閱讀:OpenCore: OSGi上部署Hibernate的四種方式【編輯推薦】