這裏先引入結論,再驗證:html
根據id直接查詢對應的數據,首先在session緩存中查找,而後在二級緩存中查找,尚未就查詢數據庫,數據庫中沒有就返回null。git
根據id直接查詢對應的數據,首先查詢session緩存,若是session中有對應對象,則直接返回該對象,若沒有就建立代理對象,實際使用數據其餘屬性(非標識符)時才查詢二級緩存和數據庫。(若是隻用到標識符,那麼便不會去作後續的查詢)github
使用load方法,hibernate認爲該id對應的對象(數據庫記錄)在數據庫中是必定存在的,因此它能夠放心的使用,它能夠放心的使用代理來延遲加載該對象。在用到對象中的其餘屬性數據時才查詢數據庫,可是萬一數據庫中不存在該記錄,那沒辦法,只能拋異常ObjectNotFoundException;數據庫
因爲session中的緩存對於hibernate來講是個至關輕量的資源,因此在load時會先查一下session緩存看看該id對應的對象是否存在,不存在則建立代理。因此若是你知道該id在數據庫中必定有對應記錄存在就可使用load方法來實現延遲加載。緩存
下面作一個簡單的測試對比:session
@Test public void getTest() { Session session = null; try { session = HibernateUtil.currentSession(); User user = session.get(User.class, 1L); System.out.println(user.getName()); }catch (Exception e) { e.printStackTrace(); }finally { HibernateUtil.closeSession(); } }
斷點調試一下:測試
如今看一下user對象spa
看一下控制檯:(已經打印出查詢語句)hibernate
接下來繼續程序:3d
@Test public void loadTest() { Session session = null; try { session = HibernateUtil.currentSession(); User user = session.load(User.class, 1L); System.out.println(user.getName()); }catch (Exception e) { e.printStackTrace(); }finally { HibernateUtil.closeSession(); } }
斷點調試一下:
看一下user
這裏發現雖然load已經執行,可是user對象是一個User_$$_jvst7b1_0
咱們打開handler 看看:(這裏只保存了id,tartget仍是爲null,說明實際並未查詢,而是先用代理類把id記錄下來)
再看一下控制檯:(並未執行select查詢)
接下來繼續運行
發現這時執行了select語句;
對比一下咱們發現差異了;
再深刻的測試,請自行驗證;
PS:源碼地址 https://github.com/JsonShare/hibernate-demo