J2EE社區(qū)

          茍有恒,何必三更起五更眠;
          最無益,只怕一日曝十日寒.
          posts - 241, comments - 318, trackbacks - 0, articles - 16

          Hibernate之中inverse與cascade的異同

          Posted on 2008-09-16 10:16 xcp 閱讀(646) 評(píng)論(0)  編輯  收藏 所屬分類: Hibernate

            最近學(xué)習(xí)hibernate,重點(diǎn)研究了hibernate屬性的inversecascade的聯(lián)系與區(qū)別。如下,是本人的心得。
             
          1、到底在哪用cascade="..."?  
             
            cascade
          屬性并不是多對(duì)多關(guān)系一定要用的,有了它只是讓我們?cè)诓迦牖騽h除對(duì)像時(shí)更方便一些,只要在cascade的源頭上插入或是刪除,所有 cascade的關(guān)系就會(huì)被自己動(dòng)的插入或是刪除。便是為了能正確的cascade,unsaved-value是個(gè)很重要的屬性。Hibernate通 過這個(gè)屬性來判斷一個(gè)對(duì)象應(yīng)該save還是update,如果這個(gè)對(duì)象的idunsaved-value的話,那說明這個(gè)對(duì)象不是 persistence   objectsaveinsert);如果id是非unsaved-value的話,那說明這個(gè)對(duì)象是persistence   object(數(shù)據(jù)庫中已存在),只要update就行了。saveOrUpdate方法用的也是這個(gè)機(jī)制。  
             
           
          2、到底在哪用inverse="ture"?  
             “set
          inverse屬性決定是否把對(duì)set的改動(dòng)反映到數(shù)據(jù)庫中去。inverse=false————反映;inverse=true————不反映”inverse屬性默認(rèn)為false
             
            inverse
          屬性默認(rèn)是false的,就是說關(guān)系的兩端都來維護(hù)關(guān)系。這個(gè)意思就是說,如有一個(gè)Student,   TeacherTeacherStudent表,StudentTeacher是多對(duì)多對(duì)多關(guān)系,這個(gè)關(guān)系由TeacherStudent這個(gè)表來表 現(xiàn)。那么什么時(shí)候插入或刪除TeacherStudent表中的記錄來維護(hù)關(guān)系呢?在用hibernate時(shí),我們不會(huì)顯示的對(duì) TeacherStudent表做操作。對(duì)TeacherStudent的操作是hibernate幫我們做的。hibernate就是看hbm文件中指 定的是""維護(hù)關(guān)系,那個(gè)在插入或刪除""時(shí),就會(huì)處發(fā)對(duì)關(guān)系表的操作。前提是""這個(gè)對(duì)象已經(jīng)知道這個(gè)關(guān)系了,就是說關(guān)系另一頭的對(duì)象已經(jīng)set 或是add""這個(gè)對(duì)象里來了。前面說過inverse默認(rèn)是false,就是關(guān)系的兩端都維護(hù)關(guān)系,對(duì)其中任一個(gè)操作都會(huì)處發(fā)對(duì)表系表的操作。當(dāng)在 關(guān)系的一頭,如Student中的bagset中用了inverse"true"時(shí),那就代表關(guān)系是由另一關(guān)維護(hù)的(Teacher)。就是說當(dāng)這插 入Student時(shí),不會(huì)操作TeacherStudent表,即使Student已經(jīng)知道了關(guān)系。只有當(dāng)Teacher插入或刪除時(shí)才會(huì)處發(fā)對(duì)關(guān)系表的 操作。所以,當(dāng)關(guān)系的兩頭都用inverse="true"是不對(duì)的,就會(huì)導(dǎo)致任何操作都不處發(fā)對(duì)關(guān)系表的操作。當(dāng)兩端都是inverse= "false"或是default值是,在代碼對(duì)關(guān)系顯示的維護(hù)也是不對(duì)的,會(huì)導(dǎo)致在關(guān)系表中插入兩次關(guān)系。  
             
           
          在一對(duì)多關(guān)系中inverse就更有意義了。在多對(duì)多中,在哪端inverse="true"效果差不多(在效率上)。但是在一對(duì)多中,如果要一方維護(hù)關(guān) 系,就會(huì)使在插入或是刪除""方時(shí)去update""方的每一個(gè)與這個(gè)""的對(duì)象有關(guān)系的對(duì)象。而如果讓""方面維護(hù)關(guān)系時(shí)就不會(huì)有update 操作,因?yàn)殛P(guān)系就是在多方的對(duì)象中的,直指插入或是刪除多方對(duì)象就行了。當(dāng)然這時(shí)也要遍歷""方的每一個(gè)對(duì)象顯示的操作修關(guān)系的變化體現(xiàn)到DB中。不管 怎樣說,還是讓""方維護(hù)關(guān)系更直觀一些。

             
          1)對(duì)one-to-many而言,改變set,會(huì)讓hibernate執(zhí)行一系列的update語句, 不會(huì)delete/insert數(shù)據(jù)
             
          2)對(duì)many-to-many而言,改變set,只修改關(guān)系表的數(shù)據(jù),不會(huì)影響many-to-many的另一方。
             
          3)雖然one-to-manymany-to-many的數(shù)據(jù)庫操作不一樣,但目的都是一個(gè):維護(hù)數(shù)據(jù)的一致性。
            
             
           
          3、cascadeinverse有什么區(qū)別?  
           
          可以這樣理解,cascade定義的是關(guān)系兩端對(duì)象到對(duì)象的級(jí)聯(lián)關(guān)系;而inverse定義的是關(guān)系和對(duì)象的級(jí)聯(lián)關(guān)系。
            inverse
          只對(duì)set+one-to-many(many-to-many)有效,對(duì)many-to-one, one-to-one無效。cascade對(duì)關(guān)系標(biāo)記都有效。

            inverse
          對(duì)集合對(duì)象整體起作用,cascade對(duì)集合對(duì)象中的一個(gè)一個(gè)元素起作用,如果集合為空,那么cascade不會(huì)引發(fā)關(guān)聯(lián)操作。
            
          比如將集合對(duì)象置為null, school.setStudentSet(null)
             inverse
          導(dǎo)致hibernate執(zhí)行:udpate STUDENT set SCHOOL_ID=null where SCHOOL_ID=?
             cascade
          則不會(huì)執(zhí)行對(duì)STUDENT表的關(guān)聯(lián)更新, 因?yàn)榧现袥]有元素。
            
          再比新增一個(gè)school, session.save(school)
             inverse
          導(dǎo)致hibernate執(zhí)行:
              for(
          對(duì)(school的每一個(gè)student ){
               udpate STUDENT set SCHOOL_ID=? where STUDENT_ID=? //
          將學(xué)生的school_id改為新的schoolid
              }
             cascade
          導(dǎo)致hibernate執(zhí)行:
              for(
          對(duì)school的每一個(gè)student ){
               session.save(aStudent); //
          對(duì)學(xué)生執(zhí)行save操作
              } 
            extends:
          如果改變集合中的部分元素(比如新增一個(gè)元素),
             inverse: hibernate
          先判斷哪些元素改變了,對(duì)改變的元素執(zhí)行相應(yīng)的sql
             cascade:
          它總是對(duì)集合中的每個(gè)元素執(zhí)行關(guān)聯(lián)操作。
              
          (在關(guān)聯(lián)操作中,hibernate會(huì)判斷操作的對(duì)象是否改變)

           
          兩個(gè)起作用的時(shí)機(jī)不同:
             cascade
          :在對(duì)主控方操作時(shí),級(jí)聯(lián)發(fā)生。
             inverse:
          flush時(shí)(commit會(huì)自動(dòng)執(zhí)行flush),對(duì)session中的所有set,hibernate判斷每個(gè)set是否有變化,
            
          對(duì)有變化的set執(zhí)行相應(yīng)的sql,執(zhí)行之前,會(huì)有個(gè)判斷:if( inverse == true ) return;可以看出cascade在先,inverse在后。
             inverse 對(duì)set + one-to-many set + many-to-many 起的作用不同。hibernate生成的sql不同。
               
          對(duì)one-to-manyhibernate對(duì)many方的數(shù)據(jù)庫表執(zhí)行update語句。
               
          對(duì)many-to-many, hibernate對(duì)關(guān)系表執(zhí)行insert/update/delte語句,注意不是對(duì)many方的數(shù)據(jù)庫表而是關(guān)系表。
              cascase
          對(duì)set都是一致的,不管one-to-many還是many-to-many。都簡(jiǎn)單地把操作傳遞到set中的每個(gè)元素。所以它總是更新many方的數(shù)據(jù)庫表。


            4
          、cascadeinverse有什么相同?
           
          這兩個(gè)屬性本身互不影響,但起的作用有些類似,都能引發(fā)對(duì)關(guān)系表的更新。

            5、 建議:只對(duì)set + many-to-many設(shè)置inverse=false,其他的標(biāo)記不考慮inverse屬性,都設(shè)為inverse=true。對(duì)cascade,一 般對(duì)many-to-one,many-to-manyconstrained=trueone-to-one 不設(shè)置級(jí)聯(lián)刪除。

               轉(zhuǎn)載于:http://www.aygfsteel.com/stme/archive/2007/02/13/99588.html




          名稱: ?4C.ESL | .↗Evon
          口號(hào): 遇到新問題?先要尋找一個(gè)方案乄而不是創(chuàng)造一個(gè)方案こ
          mail: 聯(lián)系我


          主站蜘蛛池模板: 秀山| 浦江县| 遵化市| 竹溪县| 资阳市| 福建省| 福贡县| 肃宁县| 商南县| 永嘉县| 利川市| 涟源市| 乌什县| 会宁县| 衡阳市| 临泉县| 新建县| 浦北县| 获嘉县| 卢湾区| 东乡| 从江县| 丰顺县| 南通市| 竹北市| 平利县| 万载县| 马边| 贵溪市| 外汇| 厦门市| 浪卡子县| 芒康县| 阜平县| 浦城县| 安阳县| 万州区| 高陵县| 离岛区| 娱乐| 阿图什市|