常用的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的功能,而是對hibernate-core進(jìn)行封裝江醇,使得Hibernate符合JPA的規(guī)范濒憋。
下面重點(diǎn)介紹一下hibernate-entitymanager包的主要類及實(shí)現(xiàn)。
HibernatePersistence.java嫁审,實(shí)現(xiàn)了JPA的PersistenceProvider接口,它提供 createEntityManagerFactory和createContainerEntityManagerFactory兩個(gè)方法來創(chuàng)建 EntityManagerFactory對象赖晶,這兩個(gè)方法底層都是調(diào)用的EJB3Configuration對象的 buildEntityManagerFactory方法律适,來解析JPA配置文件persistence.xml,,并創(chuàng)建 EntityManagerFactory對象遏插。
EntityManagerFactory對象的實(shí)現(xiàn)是EntityManagerFactoryImpl類捂贿,這個(gè)類有一個(gè)最重要的******* 屬性就是Hibernate的核心對象之一SessionFactory。這個(gè)類最重要的方法是createEntityManager胳嘲,來返回 EntityMnagaer對象厂僧,而sessionFactory屬性也傳入了該方法。
EntityManager對象的實(shí)現(xiàn)是EntityManagerImpl類了牛,這個(gè)類繼承自AbstractEntityManagerImpl 類颜屠,在AbstractEntityManager類中有一個(gè)抽象方法getSession來獲得Hibernate的Session對象,正是在這個(gè) Session對象的實(shí)際支持下鹰祸,EntityManagerImpl類實(shí)現(xiàn)了JPA的EntityManager接口的所有方法甫窟,并完成實(shí)際的ORM操 作。
此外蛙婴,hibernate-entitymanager包中還有QueryImpl類利用EntityManagerImpl的支持實(shí)現(xiàn)了JPA的 Query接口粗井;TransactionImpl利用EntityManagerImpl的支持實(shí)現(xiàn)了JPA的EntityTransaction接口。
至此街图,Hibernate通過hibernate-entitymanager包完成了對于JPA的全部支持工作浇衬。
這里我們要先談一下什么叫實(shí)體(Entity),按照J(rèn)PA規(guī)范,具有ORM元數(shù)據(jù)的領(lǐng)域?qū)ο缶徒凶鰧?shí)體餐济。它應(yīng)具備以下條件:
1.必須使用javax.persistence.Entity注解或XML映射文件中有對應(yīng)的元素耘擂;
2.必須具有一個(gè)不帶參數(shù)的構(gòu)造函數(shù),類不能聲明為final,方法和需要持久化的屬性也不能聲明為final;
3.如果游離態(tài)的實(shí)體對象需要以值的方式進(jìn)行傳遞(如通過Session bean的遠(yuǎn)程業(yè)務(wù)接口傳遞)絮姆,則必須實(shí)現(xiàn)Serializable接口梳星;
4.需要持久化的屬性赞赖,起訪問修飾符不能是public,它必須通過實(shí)體類方法進(jìn)行訪問冤灾。
實(shí)體的狀態(tài)
實(shí)體共有4種狀態(tài):
1前域、 新建態(tài):新創(chuàng)建的實(shí)體對象,尚未擁有持久化主鍵韵吨,沒有和一個(gè)持久化上下文關(guān)聯(lián)起來
2匿垄、 受控態(tài):已經(jīng)擁有持久化主鍵和持久化上下文建立了聯(lián)系
3、 游離態(tài):擁有持久化主鍵归粉,但尚未和持久化上下文建立聯(lián)系
4椿疗、 刪除態(tài):擁有持久化主鍵,已經(jīng)和持久化上下文建立了聯(lián)系糠悼,但已經(jīng)被安排從數(shù)據(jù)庫中刪除
下面我們對以上代碼涉及的注解進(jìn)行說明:
@Entity:將對對象標(biāo)注為一個(gè)實(shí)體類,表明需要將類持久化到數(shù)據(jù)庫當(dāng)中去届榄,一般情況類名即表名,通過name屬性顯式指定表名倔喂,如:name=”T_TEST”表示將Test保存到表T_TEST表中铝条。
@Id:對應(yīng)的屬性是表的主鍵。
@GeneratedValue:主鍵的產(chǎn)生策略席噩,通過strategy屬性進(jìn)行指定班缰,默認(rèn)情況下,JPA自動(dòng)選擇一個(gè)最適合底層數(shù)據(jù)庫的主鍵生成策略悼枢,如SqlServer對應(yīng)的identity:
mysql對應(yīng)的auto increment,在java.persistence.GenerationType中定義了幾種可以供選擇的策略:
1.? :表自動(dòng)增長字段埠忘,Oracle不支持這種方式;Identity
2.? :JPA自動(dòng)選擇合適的策略馒索,是默認(rèn)選項(xiàng)莹妒;AUTO
3.? :通過序列產(chǎn)生主鍵,通過@SequenceGenerator注解指定序列名绰上,Mysql不支持這種方式动羽。Sequence
4.? :通過表產(chǎn)生主鍵,框架借由表模擬產(chǎn)生主鍵渔期,使用該策略可以使用更易于數(shù)據(jù)庫的移植运吓。TABLE
@Colunm(name=”uname”):屬性對應(yīng)的表字段。我們并不需要指定表字段的類型疯趟,因?yàn)镴PA 會(huì)根據(jù)反射從實(shí)體屬性中獲取類型拘哨;如果是字符串類型,我們可以指定字段長度信峻,以便可以自動(dòng)生成DDL語句倦青。
@Temporal(TemporalType.DATE):如果屬性是時(shí)間類型,因?yàn)閿?shù)據(jù)表對時(shí)間類型有更嚴(yán)格的劃分盹舞,所以必須指定具體時(shí)間類型产镐,在java.persistence.TemporalType枚舉中定義了三種時(shí)間類型:
Date:等于java.sql.Date;
Time:等于java.sql.Time;
TimeStamp:等于java.sql.Timestamp隘庄。
JPA對于具有父子關(guān)系的類,對于父類必須聲明繼承實(shí)體的映射策略癣亚,對于繼承實(shí)體丑掺,java.persistence.InheritanceType定義了3種映射策略:
SINGLE_TABLE:父子類都保存在同一個(gè)表中,通過字段值進(jìn)行區(qū)分述雾。
JOINED:父子類相同的部分保存在同一個(gè)表中街州,不同的部門分開存放,通過連接不同的表獲取完整數(shù)據(jù)玻孟。
TABLE_PER_CLASS:每一個(gè)類對應(yīng)自己的表唆缴,一般不推薦采用這種方式。
下面我們來看看實(shí)際的列子是怎么運(yùn)用的黍翎。
可以看到通過字段types來區(qū)分父子類數(shù)據(jù)面徽,也是相當(dāng)方便的。至于JPA提供的關(guān)聯(lián)關(guān)系比如說一對多匣掸,多對一趟紊,多對多,也有相應(yīng)的注解進(jìn)行關(guān)聯(lián)旺聚,有興趣的朋友可以參考相關(guān)幫助文檔织阳。
以上講述的都是JPA中以注解形式進(jìn)行持久化眶蕉,下面我們來看下采用XML元數(shù)據(jù)的形式砰粹,XML元數(shù)據(jù)信息以orm.xml命名,放置在類路徑的META-INF 目錄下造挽。如果你提供了 XML 元數(shù)據(jù)描述信息碱璃,它將覆蓋實(shí)體類中的注解元數(shù)據(jù)信息。
JPA重要API
JPA接口位于javax.persistence和javax.persistence.spi兩個(gè)包中,javax.persistence包 中大部分API都是注解類饭入、EntityManager嵌器、Query等持久化操作接口。而javax.persistence.spi包中的4個(gè)API谐丢, 是JPA的服務(wù)層接口
EntityManager
實(shí)體對象由實(shí)體管理器進(jìn)行管理爽航,通過EntityManager和持久化上下文進(jìn)行交互
實(shí)體管理器有兩種:
容器類:容器型的實(shí)體管理器由容器負(fù)責(zé)試題管理器之間的協(xié)作,Java EE應(yīng)用服務(wù)器提供的就是管理型的實(shí)體管理器乾忱。
應(yīng)用程序型:實(shí)體管理器的生命周期由應(yīng)用程序控制讥珍,應(yīng)用程序通過javax.persistence.EntityManagerFactoty的creaeEntityManager創(chuàng)建EntityManager實(shí)例
EntityManager的API
void persist(Object entity)
通過persist方法,新實(shí)體實(shí)例將轉(zhuǎn)換為受控狀態(tài)窄瘟,就是說衷佃,當(dāng)persist()方法所在的事務(wù)提交時(shí),實(shí)體的數(shù)據(jù)保存到數(shù)據(jù)庫中蹄葱。
如果實(shí)體已經(jīng)被持久化氏义,那么調(diào)用persist()方法不會(huì)發(fā)生任何事情锄列。
如果對一個(gè)已經(jīng)刪除的實(shí)體調(diào)用persist()方法,刪除態(tài)的實(shí)體又轉(zhuǎn)變?yōu)槭芸貞B(tài)
如果對游離狀態(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í)體聲明為級聯(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ù)庫中
T merge(T entity)
將一個(gè)游離態(tài)的實(shí)體持久化到數(shù)據(jù)庫中,并轉(zhuǎn)換為受控態(tài)的實(shí)體
T find(Class entityClass.Object primaryKey)
以主鍵查詢實(shí)體對象紫岩,entityClass是實(shí)體的類规惰,primaryKey是主鍵值
Eg:Topic t = em.find(Topic.class,1);