首先hibernate中對象的狀態有三種:瞬態、遊離態和持久態,三種狀態轉化的方法都是經過session來調用,瞬態到持久態的方法有save()、saveOrUpdate()、get()、load();持久態到瞬態的方法有delete();遊離態到持久態的方法有update()、saveOrUpdate()、lock();持久態到遊離態的方法有:session.close()、session.evict()、session.clear()。java
hibernate的各類保存方式的區(save,persist,update,saveOrUpdte,merge,flush,lock)及 對象的三種狀態數據庫
hibernate的保存緩存
hibernate對於對象的保存提供了太多的方法,他們之間有不少不一樣,這裏細說一下,以便區別。session
在全部以前,說明一下,對於hibernate,它的對象有三種狀態,transient、persistent、detachedide
下邊是常見的翻譯辦法:spa
transient:瞬時態或者臨時態hibernate
(new DeptPo(1,」行政部」,20,」行政相關」),該po的實例和session沒有關聯,該po的實例處於transient)翻譯
persistent:持久化狀態對象
(和數據庫中記錄想影射的Po實例,它的狀態是persistent, 經過get和load等獲得的對象都是persistent)blog
detached:脫管狀態或者遊離態
1)當經過get或load方法獲得的po對象它們都處於persistent,但若是執行delete(po)時(但不能執行事務),該 po狀態就處於detached, (表示和session脫離關聯),因delete而變成遊離態能夠經過save或saveOrUpdate()變成持久態
2)當把session關閉時,session緩存中的persistent的pojo對象也變成detached。因關閉session而變成遊離態的能夠經過lock、save、update變成持久態。持久態實例能夠經過調用 delete()變成脫管狀態。經過get()或load()方法獲得的實例都是持久化狀態的。脫管狀態的實例能夠經過調用lock()或者replicate()進行持久化。save()和persist()將會引起SQL的INSERT,delete()會引起SQLDELETE,而update()或merge()會引起SQL UPDATE。對持久化(persistent)實例的修改在刷新提交的時候會被檢測到,它也會引發SQL UPDATE。saveOrUpdate()或者replicate()會引起SQLINSERT或者UPDATE
把這一對放在第一位的緣由是由於這一對是最經常使用的。
save的做用是把一個新的對象保存
update是把一個脫管狀態的對象或自由態對象(必定要和一個記錄對應)更新到數據庫
這個是比較好理解的,顧名思義,saveOrUpdate基本上就是合成了save和update,而update只是update;引用hibernate reference中的一段話來解釋他們的使用場合和區別一般下面的場景會使用update()或saveOrUpdate():
程序在第一個session中加載對象,接着把session關閉;
該對象被傳遞到表現層;
對象發生了一些改動;
該對象被返回到業務邏輯層最終到持久層;
程序建立第二session調用第二個session的update()方法持久這些改動。
saveOrUpdate(po)作下面的事:
若是該po對象已經在本session中持久化了,在本session中執行saveOrUpdate不作任何事
若是savaOrUpdate(新po)與另外一個與本session關聯的po對象擁有相同的持久化標識(identifier),拋出一個異常
org.hibernate.NonUniqueObjectException: a different object with the same identifier value was already associated with the session: [org.itfuture.www.po.Xtyhb#5]
saveOrUpdate若是對象沒有持久化標識(identifier)屬性,對其調用save() ,不然update() 這個對象
這個是最迷離的一對,表面上看起來使用哪一個都行,在hibernate reference文檔中也沒有明確的區分他們.
這裏給出一個明確的區分。(能夠跟進src看一下,雖然實現步驟相似,可是仍是有細微的差異)
主要內容區別:
1)persist把一個瞬態的實例持久化,可是並"不保證"標識符(identifier主鍵對應的屬性)被馬上填入到持久化實例中,標識符的填入可能被推遲到flush的時候。
2)save, 把一個瞬態的實例持久化標識符,及時的產生,它要返回標識符,因此它會當即執行Sql insert
比較update和merge
update的做用上邊說了,這裏說一下merge的
若是session中存在相同持久化標識(identifier)的實例,用用戶給出的對象覆蓋session已有的持久實例
(1)當咱們使用update的時候,執行完成後,會拋出異常
(2)但當咱們使用merge的時候,把處理自由態的po對象A的屬性copy到session當中處於持久態的po的屬性中,執行完成後原來是持久狀態仍是持久態,而咱們提供的A仍是自由態
這兩個的區別好理解:
update操做的是在自由態或脫管狀態(因session的關閉而處於脫管狀態)的對象//updateSQL
而flush是操做的在持久狀態的對象。
默認狀況下,一個持久狀態的對象的改動(包含set容器)是不須要update的,只要你更改了對象的值,等待hibernate flush就自動更新或保存到數據庫了。hibernate flush發生在如下幾種狀況中:
1) 調用某些查詢的和手動flush(),session的關閉、SessionFactory關閉結合
get()一個對象,把對象的屬性進行改變,把資源關閉。
2)transaction commit的時候(包含了flush)
update是把一個已經更改過的脫管狀態的對象變成持久狀態
lock是把一個沒有更改過的脫管狀態的對象變成持久狀態(針對的是因Session的關閉而處於脫管狀態的po對象(2),不能針對因delete而處於脫管狀態的po對象)
對應更改一個記錄的內容,兩個的操做不一樣:
update的操做步驟是:
(1)屬性改動後的脫管的對象的修改->調用update
lock的操做步驟是:
(2)調用lock把未修改的對象從脫管狀態變成持久狀態-->更改持久狀態的對象的內容-->等待flush或者手動flush
clear完整的清除session緩存
evcit(obj)把某個持久化對象從session的緩存中清空。
session.lock(xtyhb,LockMode.NONE);//表示直接到緩存中去找變成持久態的對象
session.lock(xtyhb,LockMode.READ);//先經過ID讀數據庫該記錄的ID看是否有該記錄,若是有接着到緩存中去找變成持久態的對象
在Hibernate中,對象有三種狀態:臨時狀態、持久狀態和遊離狀態。也叫:瞬時態(Transient)、持久態(Persistent)、脫管態(Detached)。處於持久態的對象也稱爲PO(Persistence Object),瞬時對象和脫管對象也稱爲VO(Value Object)。
臨時狀態:當new一個實體對象後,這個對象處於臨時狀態,即這個對象只是一個保存臨時數據的內存區域,若是沒有變量引用這個對象,則會被jre垃圾回收機 制回收。這個對象所保存的數據與數據庫沒有任何關係,除非經過Session的save或者SaveOrUpdate把臨時對象與數據庫關聯,並把數據插 入或者更新到數據庫,這個對象才轉換爲持久對象。
例:Emp e=new Emp(); //建立臨時對象
e.setEmpno((long) 8888);
e.setEName("mike");
...
EmpDAO d=new EmpDAO();
d.save(e); //持久化
...
持久狀態:持久化對象的實例在數據庫中有對應的記錄,並擁有一個持久化表示(ID)。對持久化對象進行delete操做後,數據庫中對應的記錄將被刪除,那麼持久化對象與數據庫記錄再也不存在對應關係,持久化對象變成臨時狀態。持久化對象被修改變動後,不會立刻同步到數據庫,知道數據庫事務提交。在同步以前,持久化對象是髒的(Dirty)。
例:
Emp e=new Emp(); //建立了臨時的對象
EmpDAO edao= new empDAO();
e=edao.findbyEmpno((long) 7786); //使對象與數據庫記錄對應,從而持久化
e.setEname("新的名字");//修改了持久化對象,使之處於 Dirty
......
edao.saveorupdate(e); //保存,可是仍然 Dirty
tran.commit(); //提交,實現與數據庫同步,再也不Dirty
......
遊離狀態:當Session進行了Close、Clear或者evict後,持久化對象雖然擁有持久化標識符和與數據庫對應記錄一致的值,可是由於 會話已經消失,對象不在持久化管理以內,因此處於遊離狀態(也叫:脫管狀態)。遊離狀態的對象與臨時狀態對象是十分類似的,只是它還含有持久化標識。
由new命令開闢內存空間的java對象,
eg. Person person = new Person("xxx", "xx"); 若是沒有變量對該對象進行引用,它將被java虛擬機回收。
瞬時對象在內存孤立存在,它是攜帶信息的載體,不和數據庫的數據有任何關聯關係,在Hibernate中,可經過session的save()或 saveOrUpdate()方法將瞬時對象與數據庫相關聯,並將數據對應的插入數據庫中,此時該瞬時對象轉變成持久化對象。
處於該狀態的對象在數據庫中具備對應的記錄,並擁有一個持久化標識。若是是用hibernate的delete()方法,對應的持久對象就變成瞬時對象,因數據庫中的對應數據已被刪除,該對象再也不與數據庫的記錄關聯。當一個session執行close()或clear()、evict()以後,持久對象變成脫管對象,此時持久對象會變成脫管對象,此時該對象雖然具備數據庫識別值,但它已不在HIbernate持久層的管理之下。
持久對象具備以下特色:
1. 和session實例關聯;
2. 在數據庫中有與之關聯的記錄。
當與某持久對象關聯的session被關閉後,該持久對象轉變爲脫管對象。當脫管對象被從新關聯到session上時,並再次轉變成持久對象。脫管對象擁有數據庫的識別值,可經過update()、saveOrUpdate()等方法,轉變成持久對象。
脫管對象具備以下特色:
1. 本質上與瞬時對象相同,在沒有任何變量引用它時,JVM會在適當的時候將它回收;
2. 比瞬時對象多了一個數據庫記錄標識值。