1. 程式人生 > >hibernate載入持久化對象的兩種方式——get、load

hibernate載入持久化對象的兩種方式——get、load

總結 font 存在 con null 同學 分析 保存 找到

一.get與load對照

在hibernate中get和load方法是依據id取得持久化對象的兩種方法。但在實際使用的過程中總會把兩者混淆,不知道什麽情況下使用get好,什麽時候使用load方法效率更高。下邊具體說一下get和load的不同,有些時候為了對照也會把find加進來。

1.從返回結果上對照:

load方式檢索不到的話會拋出org.hibernate.ObjectNotFoundException異常
get方法檢索不到的話會返回null

2.從檢索運行機制上對照:

get方法和find方法都是直接從數據庫中檢索 而load方法的運行則比較復雜首先查找session的persistent Context中是 否有緩存。假設有則直接返回 假設沒有則推斷是否是lazy。假設不是直接訪問數據庫檢索,查到記錄返回。查不到拋出異常 假設是lazy則須要建立代理對象。對象的initialized屬性為false。target屬性為null 在訪問獲得的代理對象的屬性時,檢索數據庫,假設找到記錄則把該記錄的對象拷貝到代理對象的target上。並將initialized=true,假設找不到就拋出異常。

3.根本差別說明

假設你使用load方法,hibernate覺得該id相應的對象(數據庫記錄)在數據庫中是一定存在的。所以它能夠放心的使用,它能夠放心的使用代理來延遲載入該對象。在用到對象中的其它屬性數據時才查詢數據庫,可是萬一數據庫中不存在該記錄,那沒辦法,僅僅能拋異常。

所說的load方法拋異常是指在使用 該對象的數據時,數據庫中不存在該數據時拋異常,而不是在創建這個對象時。

因為session中的緩存對於hibernate來說是個相當便宜的資源。所以在load時會先查一下session緩存看看該id相應的對象是否存在,不存在則創建代理。

所以假設你知道該id在數據庫中一定有相應記錄存在就能夠使用load方法來實現延遲載入。

對於get方法,hibernate會確認一下該id相應的數據是否存在,首先在session緩存中查找,然後在二級緩存中查找。還沒有就查數據庫,數據庫中沒有就返回null。

對於load和get方法返回類型:盡管好多書中都這麽說:“get()永遠僅僅返回實體類”。但實際上這是不對的,get方法假設在 session緩存中找到了該id相應的對象,假設剛好該對象前面是被代理過的。如被load方法使用過,或者被其它關聯對象延遲載入過,那麽返回的還是 原先的代理對象,而不是實體類對象,假設該代理對象還沒有載入實體數據(就是id以外的其它屬性數據),那麽它會查詢二級緩存或者數據庫來載入數據,可是 返回的還是代理對象。僅僅只是已經載入了實體數據。

get方法首先查詢session緩存,沒有的話查詢二級緩存。最後查詢數據庫。反而load方法創建時首先查詢session緩存,沒有就創建代理,實際使用數據時才查詢二級緩存和數據庫。


二.使用情況分析

至於何種情況使用get什麽時候使用load。我感覺最本質的還是要看要載入的持久化對象及數據量。

當要載入的對象與其它對象沒有復雜的聯系時使用get和load沒有太大的差別;當要載入的對象和其它對象之間有復雜的關聯關系。而且對象之間的關聯數量比較大的情況下使用load方法要比get方法效率高。

比方班級和學生兩個持久化對象實體,當取得班級信息時也要載入相應班級的全部學生信息。可是假設在程序中我們如今須要的知識班級信息而不須要學生信息。在這樣的情況下get方式要等到班級信息連同學生信息都載入完成才幹使用,而load方式會首先載入班級信息並保存班級與學生之間的關系,等到真正使用學生信息的時候才會去載入。所以在關系復雜而且數據量大的情況下load方式要明顯有優勢。

三.簡單總結

總之對於get和load的根本差別。一句話,hibernate對於load方法覺得該數據在數據庫中一定存在,能夠放心的使用代理來延遲載入,假設在使用過程中發現了問題,僅僅能拋異常;而對於get方法,hibernate一定要獲取到真實的數據,否則返回null。在實際使用過程中假設分不清使用哪一種,直接使用load方式就可以。

hibernate載入持久化對象的兩種方式——get、load