(9) hibernate加载持久化对象的两种方式——get、load
一.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方式即可。
總結
以上是生活随笔為你收集整理的(9) hibernate加载持久化对象的两种方式——get、load的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: (8)hibernate四种继承映射
- 下一篇: (10) Hibernate懒加载详解