Hibernate加載數據時get和load方法的區別,讓我們先看一下方法原型吧:
1.get方法
/**
* Return the persistent instance of the given entity class with the given identifier,
* or null if there is no such persistent instance. (If the instance, or a proxy for the
* instance, is already associated with the session, return that instance or proxy.)
*
* @param clazz a persistent class
* @param id an identifier
* @return a persistent instance or null
* @throws HibernateException
*/
public Object get(Class clazz, Serializable id) throws HibernateException;
2.load方法
/**
* Return the persistent instance of the given entity class with the given identifier,
* assuming that the instance exists.
* <br><br>
* You should not use this method to determine if an instance exists (use <tt>get()</tt>
* instead). Use this only to retrieve an instance that you assume exists, where non-existence
* would be an actual error.
*
* @param theClass a persistent class
* @param id a valid identifier of an existing persistent instance of the class
* @return the persistent instance or proxy
* @throws HibernateException
*/
經過比較我們可以發現:Session.load/get方法均可以根據指定的實體類和id從數據庫讀取記錄,并返回與之對應的實體對象。其區別在于:
- 如果未能發現符合條件的記錄,get方法返回null,而load方法會拋出一個ObjectNotFoundException。
- Load方法可返回實體的代理類實例,而get方法永遠直接返回實體類。
- load方法可以充分利用內部緩存和二級緩存中的現有數據,而get方法則僅僅在內部緩存中進行數據查找,如沒有發現對應數據,將越過二級緩存,直接調用SQL完成數據讀取。
Session在加載實體對象時,將經過的過程:
- 首先,Hibernate中維持了兩級緩存。第一級緩存由Session實例維護,其中保持了Session當前所有關聯實體的數據,也稱為內部緩存。而第二級緩存則存在于SessionFactory層次,由當前所有由本SessionFactory構造的Session實例共享。出于性能考慮,避免無謂的數據庫訪問,Session在調用數據庫查詢功能之前,會先在緩存中進行查詢。首先在第一級緩存中,通過實體類型和id進行查找,如果第一級緩存查找命中,且數據狀態合法,則直接返回。
- 之后,Session會在當前“NonExists”記錄中進行查找,如果“NonExists”記錄中存在同樣的查詢條件,則返回null。“NonExists”記錄了當前Session實例在之前所有查詢操作中,未能查詢到有效數據的查詢條件(相當于一個查詢黑名單列表)。如此一來,如果Session中一個無效的查詢條件重復出現,即可迅速作出判斷,從而獲得最佳的性能表現。
- 對于load方法而言,如果內部緩存中未發現有效數據,則查詢第二級緩存,如果第二級緩存命中,則返回。
- 如在緩存中未發現有效數據,則發起數據庫查詢操作(Select SQL),如經過查詢未發現對應記錄,則將此次查詢的信息在“NonExists”中加以記錄,并返回null。
- 根據映射配置和Select SQL得到的ResultSet,創建對應的數據對象。
- 將其數據對象納入當前Session實體管理容器(一級緩存)。
- 執行Interceptor.onLoad方法(如果有對應的Interceptor)。
- 將數據對象納入二級緩存。
- 如果數據對象實現了LifeCycle接口,則調用數據對象的onLoad方法。
- 返回數據對象。
============================================
Hibernate中有兩個極為相似的方法get()與load(),他們都可以通過指定的實體類與ID從數據庫中讀取數據,并返回對應的實例,但Hibernate不會搞兩個完全一樣的方法的,它們間的不同在于:
1.如果找不到符合條件的紀錄,get()方法將返回null.而load()將會報出ObjectNotFoundEcception.
2.load()方法可以返回實體的代理類實例,而get()永遠只返回實體類.
3.load()方法可以充分利用二級緩存和內部緩存的現有數據,而get()方法只在內部緩存中進行查找,如沒有發現對應數據將跳過二級緩存,直接調用SQL完成查找.
B:
呵呵,沒有說到根本點上,hibernate中get方法和load方法的根本區別在于:如果你使用load方法,hibernate認為該id對應的對象(數據庫記錄)在數據庫中是一定存在的,所以它可以放心的使用,它可以放心的使用代理來延遲加載該對象。在用到對象中的其他屬性數據時才查詢數據庫,但是萬一數據庫中不存在該記錄,那沒辦法,只能拋異常,所說的load方法拋異常是指在使用該對象的數據時,數據庫中不存在該數據時拋異常,而不是在創建這個對象時。由于session中的緩存對于hibernate來說是個相當廉價的資源,所以在load時會先查一下session緩存看看該id對應的對象是否存在,不存在則創建代理。所以如果你知道該id在數據庫中一定有對應記錄存在就可以使用load方法來實現延遲加載。
對于get方法,hibernate會確認一下該id對應的數據是否存在,首先在session緩存中查找,然后在二級緩存中查找,還沒有就查數據庫,數據庫中沒有就返回null。
對于第2點,雖然好多書中都這么說:“get()永遠只返回實體類”,但實際上這是不正確的,get方法如果在session緩存中找到了該id對應的對象,如果剛好該對象前面是被代理過的,如被load方法使用過,或者被其他關聯對象延遲加載過,那么返回的還是原先的代理對象,而不是實體類對象,如果該代理對象還沒有加載實體數據(就是id以外的其他屬性數據),那么它會查詢二級緩存或者數據庫來加載數據,但是返回的還是代理對象,只不過已經加載了實體數據。
3。胡說八道,前面已經講了,get方法首先查詢session緩存,沒有的話查詢二級緩存,最后查詢數據庫;反而load方法創建時首先查詢session緩存,沒有就創建代理,實際使用數據時才查詢二級緩存和數據庫。
總之對于get和load的根本區別,一句話,hibernate對于load方法認為該數據在數據庫中一定存在,可以放心的使用代理來延遲加載,如果在使用過程中發現了問題,只能拋異常;而對于get方法,hibernate一定要獲取到真實的數據,否則返回null。
樓主有沒有實際測試過,還是道聽途說,還是想混積分。
C:
網上有許多關于load和get方法的討論,自己做了一個小小的實驗,明確一下load和get方法的工作原理。
首先get方法沒有什么可說的,就是在Session執行此函數的時候hit一下數據庫,而load方法比較麻煩,具體的執行流程是這樣的:
java 代碼
- Session session=getSessionFactory().openSession();
- Transaction tr=session.beginTransaction();
-
-
- Student stu=(Student)session.load(Student.class, new Integer(5));
- stu.getAddress();
- tr.commit();
- session.close();
(1)查找Session所在的persistent Context中是否有緩存的persistent object,如果有則直接返回該persistent object作為stu對象;如果沒有,則需要建立代理對象,該代理對象不是我們認為的pojo,其中的代理對象的initialized屬性為false,target屬性為null。
(2)在訪問獲得的代理對象的屬性時,例如執行stu.getArress()時,因為此時的persistent Context中沒有該persistent object,所以會hit數據庫。
(3)hit數據庫時,如果在數據庫中找到該對象對應的記錄,那么用獲得的對象賦值給該代理對象的target屬性,并且將initialized屬性改為true;如果在數據庫中找不到該對象對應的記錄,那么拋出org.hibernate.ObjectNotFoundException異常。
而get方法每次執行都hit數據庫,如果沒有相對應的記錄,那么就返回null。