原文:
http://www.libelis.com/inner-index.jsp?next=jdo.htmlJDO是Java對(duì)象持久化的新的規(guī)范。JDO經(jīng)SunJava Community Process認(rèn)定。
一、歷史
JDO是對(duì)象持久化工作的綜合成果,試圖提供一個(gè)對(duì)象持久化的完全版本。JDO同時(shí)繼承于ODMG(對(duì)象數(shù)據(jù)管理小組,標(biāo)準(zhǔn)化對(duì)象數(shù)據(jù)庫的一個(gè)獨(dú)立委員會(huì))和對(duì)象關(guān)系映射工具提供商。
JSR #000012 approved in July 1999
1999-8組建的專家小組:包括Sun、Apple、BEA、IBM、Oracle、SAP、WebGain等
2000-5 完成公開評(píng)論草案
2000-6 在JavaOne上引入
2001-3 最終草案0.93
2001-5 最終草案0.96公布
2001-6 在JavaOne上啟動(dòng)
2001-11 最終草案0.98
二、目標(biāo)
定義對(duì)象級(jí)的持久化。
- 完全支持對(duì)象模型,包括引用、集合、接口、繼承
- 完全透明持久化:這使得業(yè)務(wù)對(duì)象完全獨(dú)立于任何數(shù)據(jù)庫技術(shù),這使用了一個(gè)字節(jié)碼增強(qiáng)機(jī)制(a byte-code enhancement mechanism)。
- 縮短開發(fā)周期(不再需要映射)
- 在開發(fā)小組中清晰地劃分業(yè)務(wù)人員和數(shù)據(jù)庫人員。
- 通用持久性
? JDBC限于RDBMS,JDO潛在地可以處理任何類型的數(shù)據(jù)源,包括RDBMS,ODBMS,TP監(jiān)控處理,ASCII無格式文件,xml文件,properties文件,大機(jī)上的Cobol數(shù)據(jù)庫,等
? JDO是面向大型信息系統(tǒng)的一個(gè)完全的解決方案,在這樣的系統(tǒng)中,信息存儲(chǔ)于多種異質(zhì)數(shù)據(jù)源
- 涵蓋J2EE、J2SE、J2ME的廣泛實(shí)現(xiàn)
- 強(qiáng)壯的事務(wù)模型
- 同時(shí)支持C/S和多層體系結(jié)構(gòu)
三、體系
JDO 包
PersistentCapable:擁有持久實(shí)例的類必須實(shí)現(xiàn)這個(gè)接口。管理對(duì)象生命周期。
PersistenceManager :代表了到數(shù)據(jù)源的連接。一個(gè)應(yīng)用可以打開一個(gè)或多個(gè)PersistenceManagers。
PersistenceManagerFactory:允許從數(shù)據(jù)源中得到一個(gè)PersistenceManager的實(shí)例,這個(gè)工廠也可以作為一個(gè)連接池。
Transaction:允許界定事務(wù)
Query :允許明確地聲明性地使用JDO查詢語言從數(shù)據(jù)源中獲取數(shù)據(jù)。NB:也可以通過引用之間的基本的定位,隱含地、透明地從數(shù)據(jù)源中獲取對(duì)象。
InstanceCallback:在數(shù)據(jù)庫操作中(比如before/after read, before/after write,等),定義一些鉤子,以做特殊處理(像暫時(shí)屬性的初始化)。
JDOException:JDO操作中拋出的例外。
JDO也定義了幫助類,對(duì)象標(biāo)識(shí)(由應(yīng)用或數(shù)據(jù)源管理)
JDO實(shí)現(xiàn)可以支持或不支持兼容的PersistenceManager(當(dāng)PersistenceManager是兼容的時(shí),你可以得到存儲(chǔ)于不同數(shù)據(jù)庫的對(duì)象引用)。
NB:在JDO的第一個(gè)發(fā)布版本中,并沒有嚴(yán)格地定義鎖和鎖策略。
JDO對(duì)象模型
- JDO對(duì)象模型基本上是Java的對(duì)象模型,包括所有的基本類型,引用,集合和事件接口。
- 除了系統(tǒng)定義類(system-defined classes),支持所有的字段類型(包括簡單型、可變和不變的對(duì)象類型〔immutable and mutable object types〕、用戶定義類、數(shù)組、集合、接口)。
- 支持所有的成員變量修飾符(private, public, protected, static, transient, abstract, final, synchronized, volatile)
- 除了對(duì)象狀態(tài)依賴于不可訪問的或遠(yuǎn)程對(duì)象,即繼承于java.net.SocketImpl、本地方法等,所有的用戶定義類都可以是PersistentCapable。
JDO對(duì)象生命周期
為了能夠在數(shù)據(jù)源中訪問、存儲(chǔ)對(duì)象,應(yīng)用必須首先得到一個(gè)或幾個(gè)數(shù)據(jù)源的連接。一個(gè)JDO PersistenceManager對(duì)象就代表了這樣一個(gè)連接。它可以通過PersistenceManagerFactory類得到。持久化對(duì)象必須是實(shí)現(xiàn)了PersistentCapable接口的類的實(shí)例。這樣的類可能同時(shí)擁有持久化的或臨時(shí)的(transient)實(shí)例。
為了使一個(gè)實(shí)例持久化,編程者必須調(diào)用PersistentManager的makePersistent方法。通知JDO對(duì)象為持久化或臨時(shí)的很重要,即使它們可以從JDO的行為中得到它是臨時(shí)的,比如事務(wù)管理和對(duì)象標(biāo)識(shí)。對(duì)象標(biāo)識(shí)可以由應(yīng)用管理,或者由數(shù)據(jù)源代理(這大多是在使用ODBMS實(shí)例時(shí),因?yàn)楦拍頞bjectID本身就是ODMG模型的一部分)。
JDO支持一種持久化模型,這這種模型中持久性自動(dòng)傳播到引用的對(duì)象。這種機(jī)制經(jīng)常稱為“延伸持久(persistence by reachability)”或者“傳遞持久(transitive persistence)”。這意味著一旦一個(gè)已經(jīng)持久化的對(duì)象引用了一個(gè)臨時(shí)對(duì)象,這個(gè)臨時(shí)對(duì)象自動(dòng)變成持久化的。對(duì)于JDBC編程者,這個(gè)模型可能很奇怪,但是他們會(huì)發(fā)現(xiàn)這是大多數(shù)情況下編程者希望從持久化框架中得到的支持。
例子:
pmf = (PersistenceManagerFactory) (Class.forName("com.libelis.lido.PersistenceManagerFactory").newInstance());
pmf.setConnectionDriverName("versant");
pmf.setConnectionURL(dbName);
pm = pmf.getPersistenceManager();
tx = pm.currentTransaction();
tx.begin();
Provider aProvider = new Provider("LIBeLIS");
pm.makerPersistent(aProvider); // aProvider now persists
Address anAddress = new Address("25 rue Paul Barruel", "France", "Paris");
aProvider.address = anAddress ; // anAddress now persists
tx.commit();
pm.close();
對(duì)象或者通過明確的JDO查詢結(jié)果從內(nèi)存中得到,或者通過標(biāo)準(zhǔn)的Java對(duì)象之間的導(dǎo)航(navigation)得到。
最后這個(gè)機(jī)制很強(qiáng)大,你可以把持久對(duì)象想像為在JVM堆中的一個(gè)特殊的部分,我們稱之為“客戶端緩存(client cache)”。每次當(dāng)你企圖從一個(gè)對(duì)象導(dǎo)航到另一個(gè)時(shí),如果這個(gè)對(duì)象尚不在內(nèi)存中,JDO會(huì)自動(dòng)從數(shù)據(jù)源中獲取,并把它放在緩存中。
例子
假設(shè)aProvider對(duì)象已經(jīng)裝載到內(nèi)存,但是它的Address對(duì)象還沒有。當(dāng)你寫下如下代碼時(shí):
System.out.println(aProvider.address.city);
Address對(duì)象會(huì)自動(dòng)裝載。緩存管理自動(dòng)鏈接到事務(wù)邊界(transaction boundaries),這意味著緩存將在事務(wù)結(jié)束是刷新,所有的實(shí)體都標(biāo)示為無效的。下次對(duì)象訪問它們的狀態(tài)時(shí),自動(dòng)地、透明地從數(shù)據(jù)源中再次裝載。
例子
pmf = (PersistenceManagerFactory) (Class.forName("com.libelis.lido.PersistenceManagerFactory").newInstance());
pmf.setConnectionDriverName("versant");
pmf.setConnectionURL(dbName);
pm = pmf.getPersistenceManager();
tx = pm.currentTransaction();
tx.begin();
Provider aProvider = new Provider("LIBeLIS");
pm.makerPersistent(aProvider);
Address anAddress = new Address("25 rue Paul Barruel", "France", "Paris");
aProvider.address = anAddress ;
tx.commit(); // objects are stored into the data source, client cache is discarded, references are invalid
tx.begin();
System.out.println(aProvider); // aProvider is refreshed from the data source
tx.commit();
pm.close();
每次你修改了一個(gè)對(duì)象,它在JDO緩存中對(duì)應(yīng)的實(shí)體將被標(biāo)示為“臟(dirty)”。
例子
tx.begin();
aProvider.address = aNewAdr; // aProvider is marked as dirty
tx.commit(); // aProvider will updated in the data source
在事務(wù)的結(jié)束,PersistentManager 將把所有標(biāo)示為“臟”的JDO對(duì)象(臟對(duì)象)更新到數(shù)據(jù)源中。
JDO緩存中的每一個(gè)對(duì)象都有自己的狀態(tài)(實(shí)際上它們都和一個(gè)StateManager對(duì)象相關(guān)聯(lián))。并且JDO規(guī)范規(guī)定了一大堆狀態(tài)和它們之間的過渡。如果感興趣,請(qǐng)參照J(rèn)DO規(guī)范了解詳細(xì)。
NB:這些狀態(tài)一般是JDO提供商關(guān)心的,而不是JDO編程者。
開發(fā)周期
為了達(dá)到上面提到的完全透明的持久化,JDO定義了一個(gè)稱為“增強(qiáng)(Enhancement)”的字節(jié)碼工具機(jī)制(byte-code instrumentation mechanism)。它的思想是從業(yè)務(wù)類中剔除所有的顯式的數(shù)據(jù)庫依賴代碼。和已存在的或新的數(shù)據(jù)源的映射通過外部的元數(shù)據(jù)(metadata)XML文件定義。
JDO增強(qiáng)器讀取編譯的java文件(.class文件),并且應(yīng)用元數(shù)據(jù)文件中定義的持久化規(guī)則。例示如下:

增強(qiáng)會(huì)添加到元數(shù)據(jù)文件描述的類的字節(jié)碼中:
• 實(shí)現(xiàn)PersistentCapable接口的聲明
• 該接口中聲明的方法的字節(jié)碼必須被實(shí)現(xiàn)
• 當(dāng)修改了一個(gè)屬性后該對(duì)象標(biāo)示為“臟”的代碼
• 需要時(shí)從數(shù)據(jù)源中獲取對(duì)象的代碼
• 根據(jù)元數(shù)據(jù)文件中指定的從數(shù)據(jù)源中的原始數(shù)據(jù)(raw data)到Java對(duì)象映射的代碼
NB:在專家小組中,是否要把增強(qiáng)列入JDO規(guī)范曾經(jīng)激烈爭(zhēng)論過。一些專家認(rèn)為開發(fā)者可能會(huì)被增強(qiáng)技術(shù)嚇倒。不熟悉這個(gè)技術(shù)的開發(fā)者會(huì)驚訝于字節(jié)碼增強(qiáng),這是一個(gè)事實(shí)。但是增強(qiáng)是這樣一種通用、強(qiáng)大的開發(fā)技術(shù),可以在很多案例中有效地使用。開始時(shí),每當(dāng)開發(fā)者遇到一個(gè)Bug時(shí),可能會(huì)責(zé)難增強(qiáng)器。我們強(qiáng)烈推薦增強(qiáng)器的“新手”仔細(xì)地看一下BCEL網(wǎng)站(由Source Forge開源社區(qū)提供)。在那兒,你能看到許多關(guān)于字節(jié)碼增強(qiáng)的有用信息和工具。這種機(jī)制的熟手可以在他們的Java接口中成功地使用。所有的熟手都表現(xiàn)出了興趣。
在兩層(Client/Server)結(jié)構(gòu)中整合
在傳統(tǒng)的Client/Server體系中,JDO應(yīng)用需要使用由JDO實(shí)現(xiàn)提供的PersistenceManagers把它自己連接到一個(gè)或多個(gè)數(shù)據(jù)源。

臨時(shí)對(duì)象可以持有持久對(duì)象的引用。持久對(duì)象可以持有分布在多個(gè)PersistenceManagers中的持久對(duì)象的引用。但是這不是一個(gè)必須遵從的特性。
大部分?jǐn)?shù)據(jù)庫相關(guān)的代碼都從業(yè)務(wù)對(duì)象中移去,但是事務(wù)仍需要編程者明確的劃分(使用Transaction 的begin、commit、rollback方法)。
管理運(yùn)行在內(nèi)存中的Java模型和磁盤上的數(shù)據(jù)源的物理模型之間的映射是PersistentManager的職責(zé)。當(dāng)使用ODBMS時(shí),這個(gè)映射非常直接,但使用RDBMS或更簡單的數(shù)據(jù)源時(shí)可能會(huì)非常復(fù)雜。
四、JDO和J2EE
JDO已經(jīng)設(shè)計(jì)為整合入J2EE體系。JDO依靠一個(gè)新的Connector規(guī)范來管理一個(gè)J2EE應(yīng)用和JDO容器之間的交互。在JDO規(guī)范中這個(gè)描述為管理環(huán)境(managed environments,意思是事務(wù)和連接由應(yīng)用服務(wù)器自己管理)。JDO容器和J2EE應(yīng)用服務(wù)器交互,得到數(shù)據(jù)源的連接(因?yàn)閼?yīng)用有它自己的連接池),并根據(jù)應(yīng)用服務(wù)器的JTA兼容的事務(wù)管理器執(zhí)行事務(wù)。

JDO明確的解決了J2EE體系的一個(gè)主要缺陷,就是EJB部件模型將持久性和分布性結(jié)合起來。這兩個(gè)概念是不相關(guān)的,但是EJB規(guī)范試圖簡化,迫使應(yīng)用對(duì)持久性和分布性使用相同的粒度級(jí)別。這在軟件工程的視角并不清晰,并且也不具可擴(kuò)展性。
另外一點(diǎn),EJB持久模型(CMP和BMP)過分簡化了,只能應(yīng)付有限的情況,不適合現(xiàn)實(shí)的應(yīng)用需要。感興趣的讀者可以可以很容易地J2EE的門戶中(例如the ServerSide)找到大量的EJB持久性和JDO的討論。
使用JDO,你可以使用一種非常優(yōu)雅、通用的設(shè)計(jì):Session Bean(訪問業(yè)務(wù)處理對(duì)象)自己訪問業(yè)務(wù)數(shù)據(jù)對(duì)象。這樣做,應(yīng)用仍然擁有J2EE體系所有的優(yōu)點(diǎn)(分布式、事務(wù)、連接,…),并且持久化由JDO透明、高效地管理。

JDO可以支持在異種數(shù)據(jù)源中非常復(fù)雜的映射機(jī)制,然而,EJB/CMP模型僅適用于簡單的JDBC模型。
使用JDO你沒有業(yè)務(wù)對(duì)象的復(fù)雜性的限制(然而EJB/CMP不支持繼承)。
使用JDO在你的業(yè)務(wù)數(shù)據(jù)對(duì)象中完全沒有數(shù)據(jù)庫代碼(然而使用EJB/BMP 你的業(yè)務(wù)代碼會(huì)摻雜入JDBC代碼)
業(yè)務(wù)處理對(duì)象提供了處理多個(gè)業(yè)務(wù)數(shù)據(jù)對(duì)象的方法。
業(yè)務(wù)處理對(duì)象通常是非持久化的,它們通常通過混合的JDO查詢和導(dǎo)航得到業(yè)務(wù)數(shù)據(jù)對(duì)象。
從SessionBean中剝離業(yè)務(wù)處理方法仍然很重要,因?yàn)檫@樣你的業(yè)務(wù)模型可以把眾多應(yīng)用中任何的基礎(chǔ)構(gòu)架應(yīng)用到J2EE應(yīng)用中。
資源鏈接
• Sun JDO Specification
• JDO JSR page
• Sun JDO site
• Byte-code enhancement web sites
• LIBeLIS LiDO site
• Versant enJin
摘自:http://www.javaresearch.org/article/showarticle.jsp?column=108&thread=2406