常用的ORM(Object-Relational Mapping)框架Hibernate,Sun在充分吸收現(xiàn)有的優(yōu)秀ORM框架設(shè)計(jì)思想的基礎(chǔ)上,制定了新的JPA(Java Persistence API)規(guī)范。JPA Java Persistence API,是Java EE 5的標(biāo)準(zhǔn)ORM接口,也是ejb3規(guī)范的一部分。
Hibernate和JPA的關(guān)系:
JPA可以理解為標(biāo)準(zhǔn)接口,Hibernate可以理解為其的實(shí)現(xiàn),Hibernate主要分為:hibernate-annotation、hibernate- entitymanager和hibernate-core三個(gè)組件。
Hibernate-annotation是Hibernate支持annotation方式配置的基礎(chǔ),它包括了標(biāo)準(zhǔn)的JPA annotation以及Hibernate自身特殊功能的annotation。hibernate-core是Hibernate的核心實(shí)現(xiàn),提供了Hibernate所有的核心功能。
hibernate-entitymanager實(shí)現(xiàn)了標(biāo)準(zhǔn)的JPA,可以把它看成hibernate-core和JPA之間的適配器,它并不直接提供ORM的功能,而是對(duì)hibernate-core進(jìn)行封裝,使得Hibernate符合JPA的規(guī)范。
下面重點(diǎn)介紹一下hibernate-entitymanager包的主要類(lèi)及實(shí)現(xiàn)。
HibernatePersistence.java,實(shí)現(xiàn)了JPA的PersistenceProvider接口,它提供 createEntityManagerFactory和createContainerEntityManagerFactory兩個(gè)方法來(lái)創(chuàng)建 EntityManagerFactory對(duì)象,這兩個(gè)方法底層都是調(diào)用的EJB3Configuration對(duì)象的 buildEntityManagerFactory方法,來(lái)解析JPA配置文件persistence.xml,,并創(chuàng)建 EntityManagerFactory對(duì)象。
EntityManagerFactory對(duì)象的實(shí)現(xiàn)是EntityManagerFactoryImpl類(lèi),這個(gè)類(lèi)有一個(gè)最重要的******* 屬性就是Hibernate的核心對(duì)象之一SessionFactory。這個(gè)類(lèi)最重要的方法是createEntityManager,來(lái)返回 EntityMnagaer對(duì)象,而sessionFactory屬性也傳入了該方法。
EntityManager對(duì)象的實(shí)現(xiàn)是EntityManagerImpl類(lèi),這個(gè)類(lèi)繼承自AbstractEntityManagerImpl 類(lèi),在AbstractEntityManager類(lèi)中有一個(gè)抽象方法getSession來(lái)獲得Hibernate的Session對(duì)象,正是在這個(gè) Session對(duì)象的實(shí)際支持下,EntityManagerImpl類(lèi)實(shí)現(xiàn)了JPA的EntityManager接口的所有方法,并完成實(shí)際的ORM操 作。
此外,hibernate-entitymanager包中還有QueryImpl類(lèi)利用EntityManagerImpl的支持實(shí)現(xiàn)了JPA的 Query接口;TransactionImpl利用EntityManagerImpl的支持實(shí)現(xiàn)了JPA的EntityTransaction接口。
至此,Hibernate通過(guò)hibernate-entitymanager包完成了對(duì)于JPA的全部支持工作。
這里我們要先談一下什么叫實(shí)體(Entity),按照J(rèn)PA規(guī)范,具有ORM元數(shù)據(jù)的領(lǐng)域?qū)ο缶徒凶鰧?shí)體。它應(yīng)具備以下條件:
1.必須使用javax.persistence.Entity注解或XML映射文件中有對(duì)應(yīng)的元素;
2.必須具有一個(gè)不帶參數(shù)的構(gòu)造函數(shù),類(lèi)不能聲明為final,方法和需要持久化的屬性也不能聲明為final;
3.如果游離態(tài)的實(shí)體對(duì)象需要以值的方式進(jìn)行傳遞(如通過(guò)Session bean的遠(yuǎn)程業(yè)務(wù)接口傳遞),則必須實(shí)現(xiàn)Serializable接口;
4.需要持久化的屬性,起訪問(wèn)修飾符不能是public,它必須通過(guò)實(shí)體類(lèi)方法進(jìn)行訪問(wèn)。
實(shí)體的狀態(tài)
實(shí)體共有4種狀態(tài):
1、 新建態(tài):新創(chuàng)建的實(shí)體對(duì)象,尚未擁有持久化主鍵,沒(méi)有和一個(gè)持久化上下文關(guān)聯(lián)起來(lái)
2、 受控態(tài):已經(jīng)擁有持久化主鍵和持久化上下文建立了聯(lián)系
3、 游離態(tài):擁有持久化主鍵,但尚未和持久化上下文建立聯(lián)系
4、 刪除態(tài):擁有持久化主鍵,已經(jīng)和持久化上下文建立了聯(lián)系,但已經(jīng)被安排從數(shù)據(jù)庫(kù)中刪除

下面我們對(duì)以上代碼涉及的注解進(jìn)行說(shuō)明:
@Entity:將對(duì)對(duì)象標(biāo)注為一個(gè)實(shí)體類(lèi),表明需要將類(lèi)持久化到數(shù)據(jù)庫(kù)當(dāng)中去,一般情況類(lèi)名即表名,通過(guò)name屬性顯式指定表名,如:name=”T_TEST”表示將Test保存到表T_TEST表中。
@Id:對(duì)應(yīng)的屬性是表的主鍵。
@GeneratedValue:主鍵的產(chǎn)生策略,通過(guò)strategy屬性進(jìn)行指定,默認(rèn)情況下,JPA自動(dòng)選擇一個(gè)最適合底層數(shù)據(jù)庫(kù)的主鍵生成策略,如SqlServer對(duì)應(yīng)的identity:
mysql對(duì)應(yīng)的auto increment,在java.persistence.GenerationType中定義了幾種可以供選擇的策略:
1.? :表自動(dòng)增長(zhǎng)字段,Oracle不支持這種方式;Identity
2.? :JPA自動(dòng)選擇合適的策略,是默認(rèn)選項(xiàng);AUTO
3.? :通過(guò)序列產(chǎn)生主鍵,通過(guò)@SequenceGenerator注解指定序列名,Mysql不支持這種方式。Sequence
4.? :通過(guò)表產(chǎn)生主鍵,框架借由表模擬產(chǎn)生主鍵,使用該策略可以使用更易于數(shù)據(jù)庫(kù)的移植。TABLE
@Colunm(name=”uname”):屬性對(duì)應(yīng)的表字段。我們并不需要指定表字段的類(lèi)型,因?yàn)镴PA 會(huì)根據(jù)反射從實(shí)體屬性中獲取類(lèi)型;如果是字符串類(lèi)型,我們可以指定字段長(zhǎng)度,以便可以自動(dòng)生成DDL語(yǔ)句。
@Temporal(TemporalType.DATE):如果屬性是時(shí)間類(lèi)型,因?yàn)閿?shù)據(jù)表對(duì)時(shí)間類(lèi)型有更嚴(yán)格的劃分,所以必須指定具體時(shí)間類(lèi)型,在java.persistence.TemporalType枚舉中定義了三種時(shí)間類(lèi)型:
Date:等于java.sql.Date;
Time:等于java.sql.Time;
TimeStamp:等于java.sql.Timestamp。
JPA對(duì)于具有父子關(guān)系的類(lèi),對(duì)于父類(lèi)必須聲明繼承實(shí)體的映射策略,對(duì)于繼承實(shí)體,java.persistence.InheritanceType定義了3種映射策略:
SINGLE_TABLE:父子類(lèi)都保存在同一個(gè)表中,通過(guò)字段值進(jìn)行區(qū)分。
JOINED:父子類(lèi)相同的部分保存在同一個(gè)表中,不同的部門(mén)分開(kāi)存放,通過(guò)連接不同的表獲取完整數(shù)據(jù)。
TABLE_PER_CLASS:每一個(gè)類(lèi)對(duì)應(yīng)自己的表,一般不推薦采用這種方式。
下面我們來(lái)看看實(shí)際的列子是怎么運(yùn)用的。

可以看到通過(guò)字段types來(lái)區(qū)分父子類(lèi)數(shù)據(jù),也是相當(dāng)方便的。至于JPA提供的關(guān)聯(lián)關(guān)系比如說(shuō)一對(duì)多,多對(duì)一,多對(duì)多,也有相應(yīng)的注解進(jìn)行關(guān)聯(lián),有興趣的朋友可以參考相關(guān)幫助文檔。
以上講述的都是JPA中以注解形式進(jìn)行持久化,下面我們來(lái)看下采用XML元數(shù)據(jù)的形式,XML元數(shù)據(jù)信息以orm.xml命名,放置在類(lèi)路徑的META-INF 目錄下。如果你提供了 XML 元數(shù)據(jù)描述信息,它將覆蓋實(shí)體類(lèi)中的注解元數(shù)據(jù)信息。
JPA重要API
JPA接口位于javax.persistence和javax.persistence.spi兩個(gè)包中,javax.persistence包 中大部分API都是注解類(lèi)、EntityManager、Query等持久化操作接口。而javax.persistence.spi包中的4個(gè)API, 是JPA的服務(wù)層接口
EntityManager
實(shí)體對(duì)象由實(shí)體管理器進(jìn)行管理,通過(guò)EntityManager和持久化上下文進(jìn)行交互
實(shí)體管理器有兩種:
容器類(lèi):容器型的實(shí)體管理器由容器負(fù)責(zé)試題管理器之間的協(xié)作,Java EE應(yīng)用服務(wù)器提供的就是管理型的實(shí)體管理器。
應(yīng)用程序型:實(shí)體管理器的生命周期由應(yīng)用程序控制,應(yīng)用程序通過(guò)javax.persistence.EntityManagerFactoty的creaeEntityManager創(chuàng)建EntityManager實(shí)例
EntityManager的API
void persist(Object entity)
通過(guò)persist方法,新實(shí)體實(shí)例將轉(zhuǎn)換為受控狀態(tài),就是說(shuō),當(dāng)persist()方法所在的事務(wù)提交時(shí),實(shí)體的數(shù)據(jù)保存到數(shù)據(jù)庫(kù)中。
如果實(shí)體已經(jīng)被持久化,那么調(diào)用persist()方法不會(huì)發(fā)生任何事情。
如果對(duì)一個(gè)已經(jīng)刪除的實(shí)體調(diào)用persist()方法,刪除態(tài)的實(shí)體又轉(zhuǎn)變?yōu)槭芸貞B(tài)
如果對(duì)游離狀態(tài)的實(shí)體執(zhí)行persist()操作,拋出IllegalArgumentException
一個(gè)實(shí)體調(diào)用persist()方法后,所有與之關(guān)聯(lián)的實(shí)體,都將執(zhí)行持久化操作
void remove(Object entity)
刪除一個(gè)受控態(tài)的實(shí)體。
如果實(shí)體聲明為級(jí)聯(lián)刪除(cascade=REMOVE或者cascade=ALL),被關(guān)聯(lián)的實(shí)體也會(huì)被刪除
在一個(gè)新建態(tài)或刪除態(tài)的實(shí)體上調(diào)用remove()方法,將被忽略
在游離態(tài)的實(shí)體上調(diào)用remove()方法,將拋出IllegalArgumentException,相關(guān)事務(wù)將回滾
void flush()
將受控態(tài)的實(shí)體數(shù)據(jù)同步到數(shù)據(jù)庫(kù)中
T merge(T entity)
將一個(gè)游離態(tài)的實(shí)體持久化到數(shù)據(jù)庫(kù)中,并轉(zhuǎn)換為受控態(tài)的實(shí)體
T find(Class entityClass.Object primaryKey)
以主鍵查詢(xún)實(shí)體對(duì)象,entityClass是實(shí)體的類(lèi),primaryKey是主鍵值
Eg:Topic t = em.find(Topic.class,1);