<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    Java,J2EE,Weblogic,Oracle

    java項目隨筆
    隨筆 - 90, 文章 - 6, 評論 - 61, 引用 - 0
    數據加載中……

    Hibernate里load和get方法的區別

     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 代碼
     
    1. Session session=getSessionFactory().openSession();  
    2. Transaction tr=session.beginTransaction();  
    3. //Student stu2=(Student)session.get(Student.class, new Integer(5));  
    4. //if(session.contains(stu2)) System.out.println("stu2 in the session");  
    5. Student stu=(Student)session.load(Student.classnew Integer(5));  
    6. stu.getAddress();  
    7. tr.commit();  
    8. 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。




    )當數據庫不存在對應ID數據時,調用load()方法將會拋出ObjectNotFoundException異常,get()方法將返回null.下面是一個簡單的測試數據庫中沒有id為2的person

    2)也就是延時加載的區別。load的方法默認要加載的對象是存在數據庫中的,返回的是一個代理對象而不是一個真正的類實例,當用到具體與數據庫有關的數據時候才查詢數據庫,而get方法直接查詢數據庫,返回類的實例。例如上面的例子,運行到(1)直接輸出sql語句,而(3)卻沒有輸出,在(5)輸出了一條sql語句,由于所查詢的對象不存在拋出ObjectNotFoundException異常。關于類的代理對象可以通過debug調試的到。但是如果在內部內存中存在所要查詢的對象,無論對象是代理的(load出來的)還是類的實例(get出來的)都會返回內存中的形態。

    3)第三點區別就是:“,get方法首先查詢session緩存,沒有的話查詢二級緩存,最后查詢數據庫;反而load方法創建時首先查詢session緩存,沒有就創建代理,實際使用數據時才查詢二級緩存和數據庫。”這是從別的地方聽說的。

    posted on 2009-03-20 12:49 龔椿深 閱讀(1274) 評論(0)  編輯  收藏


    只有注冊用戶登錄后才能發表評論。


    網站導航:
     
    主站蜘蛛池模板: 性xxxxx免费视频播放| 亚洲人成免费网站| 免费在线不卡视频| 亚洲国产美女精品久久久| 国产免费不卡v片在线观看| 亚洲国产精品一区二区久| 鲁大师在线影院免费观看| 亚洲AV日韩精品久久久久久| 久久久久国产精品免费看| 麻豆亚洲AV永久无码精品久久| 91大神免费观看| 亚洲国产综合第一精品小说| 国产va免费精品观看精品| 亚洲国产精品18久久久久久 | 黄页网站在线免费观看| 国产精品色午夜视频免费看| 边摸边吃奶边做爽免费视频网站| heyzo亚洲精品日韩| 国产精品免费久久久久久久久| 亚洲综合无码精品一区二区三区| 永久免费AV无码网站国产| 亚洲精品视频免费看| 最新中文字幕电影免费观看| 美女被羞羞网站免费下载| 亚洲日韩欧洲乱码AV夜夜摸| 久久九九AV免费精品| 亚洲免费黄色网址| avtt亚洲天堂| 久久w5ww成w人免费| 亚洲国产AV无码一区二区三区| 2048亚洲精品国产| 37pao成人国产永久免费视频| 色婷婷六月亚洲综合香蕉| 自拍偷自拍亚洲精品情侣| 久久午夜免费视频| 一级a性色生活片久久无少妇一级婬片免费放 | 亚洲精品熟女国产| 青青草国产免费久久久下载 | 中文字幕免费人成乱码中国| 亚洲精品欧洲精品| 免费99热在线观看|