近年來(lái)ORM(Object-Relational Mapping)對(duì)象關(guān)系映射,即實(shí)體對(duì)象和數(shù)據(jù)庫(kù)表的映射)技術(shù)市場(chǎng)人聲音鼎沸,異常熱鬧, 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ī)范的一部分。 那么什么是JPA呢?JPA是通過JDK5.0注解或XML描述對(duì)象-關(guān)系表的映射關(guān)系,并將運(yùn)行期實(shí)體對(duì)象持久化到數(shù)據(jù)庫(kù)中去。 Hibernate與JPA的關(guān)系及其實(shí)現(xiàn)機(jī)制 JPA和Hibernate之間的關(guān)系,可以簡(jiǎn)單的理解為JPA是標(biāo)準(zhǔn)接口,Hibernate是實(shí)現(xiàn)。那么Hibernate是如何實(shí)現(xiàn)與JPA的這種關(guān)系的呢。Hibernate主要是通過三個(gè)組件來(lái)實(shí)現(xiàn)的,及hibernate-annotation、hibernate-entitymanager和hibernate-core。 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包的主要類及實(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類,這個(gè)類有一個(gè)最重要的*******屬性就是Hibernate的核心對(duì)象之一SessionFactory。這個(gè)類最重要的方法是createEntityManager,來(lái)返回EntityMnagaer對(duì)象,而sessionFactory屬性也傳入了該方法。 EntityManager對(duì)象的實(shí)現(xiàn)是EntityManagerImpl類,這個(gè)類繼承自AbstractEntityManagerImpl類,在AbstractEntityManager類中有一個(gè)抽象方法getSession來(lái)獲得Hibernate的Session對(duì)象,正是在這個(gè)Session對(duì)象的實(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包完成了對(duì)于JPA的全部支持工作。
這里我們要先談一下什么叫實(shí)體(Entity),按照J(rèn)PA規(guī)范,具有ORM元數(shù)據(jù)的領(lǐng)域?qū)ο缶徒凶鰧?shí)體。它應(yīng)具備一下條件: 實(shí)體的狀態(tài) 實(shí)體共有4種狀態(tài): 1、 新建態(tài):新創(chuàng)建的實(shí)體對(duì)象,尚未擁有持久化主鍵,沒有和一個(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ù)中刪除
下面我們來(lái)嘗試對(duì)一個(gè)域?qū)ο筮M(jìn)行JPA注解,使其成為一個(gè)實(shí)體類: @Entity(name=”T_TEST”) public class Test implements Serializable{ @Id @GeneratedValue(strategy=GenerationType.TABLE) @Column(name=” id”) ******* int testId; @Column(name=”uname”,length=100) ******* String uname; @Column(name=”password”) ******* String password; @Column(name=”time”) @Temporal(TemporalType.Date) ******* Date loginTime; //省略get/setter方法 }
@Entity:將領(lǐng)域?qū)ο髽?biāo)注為一個(gè)實(shí)體類,表示該類需要持久化到數(shù)據(jù)庫(kù)中,默認(rèn)情況下類名即表名,通過name屬性顯式指定表名,如:name=”T_TEST”表示將Test保存到表T_TEST表中。 @Id:對(duì)應(yīng)的屬性是表的主鍵 @GeneratedValue:主鍵的產(chǎn)生策略,通過strategy屬性進(jìn)行指定,默認(rèn)情況下,JPA自動(dòng)選擇一個(gè)最適合底層數(shù)據(jù)庫(kù)的主鍵生成策略,如SqlServer對(duì)應(yīng)的identity:
@Colunm(name=”uname”):屬性對(duì)應(yīng)的表字段。我們并不需要指定表字段的類型,因?yàn)镴PA 會(huì)根據(jù)反射從實(shí)體屬性中獲取類型;如果是字符串類型,我們可以指定字段長(zhǎng)度,以便可以自動(dòng)生成DDL語(yǔ)句。 @Temporal(TemporalType.DATE):如果屬性是時(shí)間類型,因?yàn)閿?shù)據(jù)表對(duì)時(shí)間類型有更嚴(yán)格的劃分,所以必須指定具體時(shí)間類型,在java.persistence.TemporalType枚舉中定義了三種時(shí)間類型:
SINGLE_TABLE:父子類都保存在同一個(gè)表中,通過字段值進(jìn)行區(qū)分。 JOINED:父子類相同的部分保存在同一個(gè)表中,不同的部門分開存放,通過連接不同的表獲取完整數(shù)據(jù)。 TABLE_PER_CLASS:每一個(gè)類對(duì)應(yīng)自己的表,一般不推薦采用這種方式。
下面我們來(lái)看看實(shí)際的列子是怎么運(yùn)用的。 @Entity(name=”test”) @Inheritance(strategy=InheritanceType.SINGLE_TABLE)//指定繼承策略 @DiscriminatorColumn(name=”types”,discriminatorType=DiscriminatorType.INTEGER,length=1)//指定區(qū)分字段為types,類型為Integer長(zhǎng)度為1 @DiscriminatorValue(value=”1”)//對(duì)應(yīng)具體實(shí)體的值 public class Test implements Serializable{ ….. }
@Entity @DiscriminatorValue(value=”2”) public class Child extends Test{ //如果我們不希望JPA將該屬性持久化到數(shù)據(jù)庫(kù),則采用該注解 @Transient ******* String tempStr; @Lob //lob類型的字段 @Basic(fetch=FetchType.Lazy) //采用延遲加載,F(xiàn)etchType.EAGER不采用 @Column(name=”postattach”,columnDefinition=”LONGTEXT NOT NULL”)對(duì)應(yīng)字段類型 ******* String postAttach; }
<?xml version=”1.0” encoding=”UTF-8”?> <entity-mappings xmlns=”http://java.sun.com/xml/ns/persistence/orm” xmlns=”http://www.w3.org/2001/XMLSchema-instance” xsi:schemaLocation=” http://java.sun.com/xml/ns/persistence/orm http://java.sun.com/xml/ns/persistence/orm_1_0.xsd” version=”1.0”> <package>com.test</package> <entity class=”Test”> <table name=”test”/> <attributes> <id name=”id”> <column name=”id”/> <generated-value strategy=”TABLE”/> </id> <basic name=”uname”> <column name=”uname” length=”30”/> </basic> <basic name=”logintime”> <column name=” logintime”/> <temporal>DATE </temporal> </basic> </attributes> </entity> <entity-mappings> 可以看到JPA元數(shù)據(jù)采用XML形式也是相當(dāng)簡(jiǎn)單易懂的。
JPA重要API JPA接口位于javax.persistence和javax.persistence.spi兩個(gè)包中,javax.persistence包中大部分API都是注解類、EntityManager、Query等持久化操作接口。而javax.persistence.spi包中的4個(gè)API,是JPA的服務(wù)層接口 EntityManager 實(shí)體對(duì)象由實(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),就是說(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) 以主鍵查詢實(shí)體對(duì)象,entityClass是實(shí)體的類,primaryKey是主鍵值 Eg:Topic t = em.find(Topic.class,1); |
|
來(lái)自: wjw_595 > 《Hibernate》