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

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

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

    posts - 70,comments - 408,trackbacks - 0

          Hibernate的映射關(guān)聯(lián)關(guān)系和我們現(xiàn)實(shí)世界里事物的關(guān)聯(lián)關(guān)系一樣.比如在UML語(yǔ)言中,以客戶Customer和訂單Order的關(guān)系為例.一個(gè)客戶可以發(fā)送多個(gè)訂單,而一個(gè)訂單只能屬于一個(gè)客戶,這是一對(duì)多的關(guān)聯(lián),因此可以成為單向關(guān)聯(lián).如果同時(shí)包含了兩兩種關(guān)聯(lián)關(guān)系,就成為雙向關(guān)聯(lián).在關(guān)系數(shù)據(jù)庫(kù)中只有外鍵參照主鍵的關(guān)系.所以關(guān)系數(shù)據(jù)庫(kù)實(shí)際上至支持一對(duì)一,或一對(duì)多的單向關(guān)系.在類于類之間的關(guān)系中.要算多對(duì)一關(guān)系和數(shù)據(jù)庫(kù)中的外鍵參照主鍵關(guān)系最匹配了.因此如果使用單向關(guān)聯(lián)從訂單到客戶的多對(duì)一單向關(guān)聯(lián),在訂單類中就要定義一個(gè)客戶的屬性.表示這個(gè)訂單屬于哪個(gè)客戶,而客戶類就無(wú)需定義存放訂單的集合屬性了.下面寫一個(gè)簡(jiǎn)單的例子.
    //首先定義客戶類
    public class Customer implements Sreializable {
       private Long id;
       private String name;
       //省略屬性的訪問(wèn)方法
    }
    //然后定義訂單類
    public class Order implements Sreializable {
       private Long id;
       private String orderName;
       private Customer customer;
       //省略屬性的訪問(wèn)方法,要注意的是Customer的訪問(wèn)方法.
    }
    Customer類的所有屬性和CUSTOMERS表的所有屬性一一對(duì)應(yīng),創(chuàng)建起來(lái)就比較簡(jiǎn)單了.下面主要看一下Order類的映射文件.
    <property name="orderName" type="string">
       <column name="ORDER_NAME" length="15"/>
    </property>
    因?yàn)閏ustomer屬性是是Customer類型,而ORDERS表的CUSTOMER_ID是整數(shù)類型,是不匹配的.所以我們不能用普通的<property>元素來(lái)定義,而我們需要使用<many-to-one>元素來(lái)配置了.
    <many-to-one name="customer" column="CUSTOMER_ID" class="包名.Customer" not-null="true"/>
    <many-to-one>元素負(fù)責(zé)建立Order訂單類的customer屬性和數(shù)據(jù)庫(kù)中的CUSTOMER_ID外鍵字段之間的映射.
    name:設(shè)定映射文件的屬性名
    column:設(shè)定和持久化類對(duì)應(yīng)的表的外鍵名
    class:設(shè)定持久化類的屬性的類型,這里指定具體的類,也就是主鍵存在的類
    not-null:設(shè)定為true表示customer屬性不允許為null,默認(rèn)是false,這個(gè)屬性會(huì)影響到bhm2ddl工具,會(huì)為ORDERS表的CUSTOMER_ID外鍵設(shè)置為不允許空的約束,但是不會(huì)影響到hbm2java工具生長(zhǎng)java源代碼.此外還會(huì)影響到Hibernate運(yùn)行時(shí)的行為,在保存Order對(duì)象的時(shí)候會(huì)檢查customer屬性是否為null.用hbm2ddl編譯之后得到的數(shù)據(jù)庫(kù)文件如下:

    create table CUSTOMERS (
       ID bigint not null,
       NAME varchar(15),
       primary key (ID)
    );
    create table ORDERS (
       ID bigint not null,
       ORDER_NUMBER varchar(15),
       CUSTOMER_ID bigint not null,
       primary key (ID)
    );
    alter table ORDERS add index FK8B7256E516B4891C (CUSTOMER_ID),
    add constraint FK8B7256E516B4891C foreign key (CUSTOMER_ID) references CUSTOMERS (ID);
    看到結(jié)果我們可以簡(jiǎn)單的把<many-to-one>理解為在數(shù)據(jù)庫(kù)中,創(chuàng)建外鍵的作用.上邊這個(gè)例子就簡(jiǎn)單的演示了Hibernate映射的一對(duì)一關(guān)聯(lián)關(guān)系,至于一對(duì)多的關(guān)聯(lián)關(guān)系比這個(gè)稍微復(fù)雜一點(diǎn).而且可以看出,當(dāng)Hibernate持久化一個(gè)臨時(shí)對(duì)象的時(shí)候,在默認(rèn)的情況下它不會(huì)自動(dòng)持久化關(guān)聯(lián)其他臨時(shí)對(duì)象,而是會(huì)拋出TransientObjectException異常.如果希望Hibernate持久化對(duì)象的時(shí)候也自動(dòng)持久化說(shuō)關(guān)聯(lián)的對(duì)象,就要把<many-to-one>元素的cascade屬性設(shè)置為save-update,表示級(jí)聯(lián)操作的意思,cascade屬性的默認(rèn)值為none.當(dāng)這個(gè)屬性設(shè)置OK了.數(shù)據(jù)庫(kù)就實(shí)現(xiàn)了級(jí)聯(lián)保存更新的操作.
          在類和類之間建好了關(guān)聯(lián)關(guān)系之后,就可以方便的從一個(gè)對(duì)象得到它關(guān)聯(lián)的對(duì)象.例如Customer customer=order.getCustomer();這樣獲得的了Customer對(duì)象了.但是如果想獲得所有屬于Customer客戶的Order訂單對(duì)象,就涉及到了一對(duì)多雙向關(guān)聯(lián)了.在內(nèi)存中,從一個(gè)對(duì)象導(dǎo)航都另一個(gè)對(duì)象要比從數(shù)據(jù)庫(kù)中通過(guò)一個(gè)字段查詢另一個(gè)字段快的多的多,但是也給編程的時(shí)候帶來(lái)了麻煩,隨意修改一個(gè)對(duì)象就可能牽一發(fā)而動(dòng)全身,所以說(shuō)雙向的關(guān)聯(lián)比較復(fù)雜,但是類和類之間到底建立單向還是雙向關(guān)聯(lián),這個(gè)要根據(jù)業(yè)務(wù)需求來(lái)決定.比如說(shuō)業(yè)務(wù)需求根據(jù)指定客戶查詢客戶所有訂單,根據(jù)指定的訂單,查詢出發(fā)這個(gè)訂單的客戶.這個(gè)時(shí)候我們不妨用多對(duì)一雙向關(guān)聯(lián)處理.其實(shí)上邊的例子的映射文件已經(jīng)簡(jiǎn)歷了客戶和訂單之間的一對(duì)多雙向關(guān)聯(lián)關(guān)系,只不過(guò)要在客戶類中加一個(gè)集合的屬性:
    private set orders = new HashSet();
    public set getOrders() {
       return orders;
    }
    public void setOrders(Set orders) {
       this.orders = orders;
    }
    有了orders屬性,客戶就可以通過(guò)getOrders()方法或者客戶的全部訂單了,Hibernate在定義這個(gè)集合屬性的時(shí)候必須聲明為接口類型,但是不光光是Set還有Map和List,這樣可以提高程序的強(qiáng)壯性,就是說(shuō)set方法接受的對(duì)象只要是實(shí)現(xiàn)了Set接口就OK.避免出現(xiàn)null值的現(xiàn)象.這里要注意的是hbm2java工具生成類的集合屬性的代碼時(shí),不會(huì)給它初始化一個(gè)集合對(duì)象的實(shí)例,這里我們需要自己手動(dòng)修改,當(dāng)然不修改也是可以的.接下來(lái)還要在customer.hbm.xml映射文件里映射集合類型的orders屬性,當(dāng)然這個(gè)和order表的的<many-to-one>同理,所以不能通過(guò)普通的<property>元素來(lái)設(shè)置屬性和字段的映射關(guān)系.要使用<set>元素來(lái)設(shè)置:
    <set name="orders" cascade="save-update">
    <key column="CUSTOMER_ID">
    <one-to-many class="包名.Order">
    </set>
    name:設(shè)定類的屬性名
    cascade:設(shè)置為save-update表示級(jí)聯(lián)保存更新,當(dāng)保存或更新Customer類的時(shí)候會(huì)級(jí)聯(lián)保存更新跟它關(guān)聯(lián)的Order類.
    <key>元素是用來(lái)設(shè)定跟持久化類關(guān)聯(lián)的類的外鍵
    <one-to-many>元素看起來(lái)很熟悉,哦是設(shè)置外鍵的元素反過(guò)來(lái)了.這里它是用來(lái)設(shè)置所關(guān)聯(lián)的持久化類的.這里設(shè)置為和客戶關(guān)聯(lián)的訂單Order類,這里表明這個(gè)屬性里要存放一組Order類型的對(duì)象.
    這個(gè)<set>元素是表示orders屬性聲明為set類型.
    <set>元素還有一個(gè)inverse屬性,這個(gè)方法主要是在給已存在數(shù)據(jù)庫(kù)中的字段建立關(guān)聯(lián)的時(shí)候很有用.就是說(shuō)當(dāng)我們獲得數(shù)據(jù)庫(kù)中的兩個(gè)表的兩條記錄的對(duì)象customer客戶對(duì)象和order訂單對(duì)象(映射文件已經(jīng)建立了他們類和類之間的關(guān)聯(lián),但外鍵的值為null的情況下)然后我們想建立這個(gè)客戶對(duì)象和訂單對(duì)象之間的關(guān)聯(lián),我們要先調(diào)用order.setCustomer(customer);然后在調(diào)用customer.getOrder().add(order);在Hibernate自動(dòng)清理緩存的持久化對(duì)象的時(shí)候會(huì)提交兩條SQL語(yǔ)句.進(jìn)行了兩個(gè)update操作.但是實(shí)際上只修改了一條記錄.重復(fù)的執(zhí)行SQL語(yǔ)句是會(huì)降低系統(tǒng)的運(yùn)行效率的,當(dāng)把inverse屬性設(shè)置為true的時(shí)候,同樣的操作就會(huì)合并到一條SQL語(yǔ)句執(zhí)行了,inverse默認(rèn)為false;
          級(jí)聯(lián)刪除就很簡(jiǎn)單了,把cascade屬性設(shè)置為delete,如果你刪除了一個(gè)客戶,程序就會(huì)先執(zhí)行刪除這個(gè)客戶全部的訂單的SQL語(yǔ)句,然后在刪除這個(gè)客戶,所謂刪除一個(gè)持久化對(duì)象不是在內(nèi)存中刪除這個(gè)對(duì)象,而是刪除數(shù)據(jù)庫(kù)中相關(guān)的記錄,這個(gè)對(duì)象依然在內(nèi)存中,只不過(guò)由持久化狀態(tài)轉(zhuǎn)為臨時(shí)狀態(tài),當(dāng)這個(gè)對(duì)象的引用消失后,這個(gè)對(duì)象會(huì)被垃圾回收.但是如果我又想級(jí)聯(lián)刪除,還想級(jí)聯(lián)保存,更新的時(shí)候應(yīng)該怎么辦呢?這個(gè)時(shí)候我們將cascade屬性設(shè)置為all-delete-orphan就OK了.非常簡(jiǎn)單明了.我們還可以通過(guò)持久化類的customer.getOrder().rumove(order);解除關(guān)聯(lián).這里的操作表示獲得客戶訂單的集合對(duì)象,然后從集合對(duì)象中刪除order的訂單,其實(shí)這種操作的意義不大,當(dāng)我們不需要的這個(gè)訂單的時(shí)候完全可以刪除它,解除關(guān)聯(lián)之后如果設(shè)置了級(jí)聯(lián)刪除屬性,這個(gè)無(wú)用的記錄也是要被刪除的.其實(shí)解除關(guān)聯(lián)就是把外鍵設(shè)為null.通常我們的外鍵都要約束不可以為空.
          映射關(guān)聯(lián)還有一種多對(duì)多的關(guān)聯(lián),是一種自身關(guān)聯(lián)關(guān)系.就是同一張表.自己和自己的關(guān)聯(lián).比如說(shuō)一張人表,地球人是人,美國(guó)人,中國(guó)人,日本人都屬于地球人,中國(guó)人有分北京人,山東人.日本人也有下一級(jí)的比如東京人.下面設(shè)想如果日本人被消滅掉了,那么東京人也應(yīng)該都被沒有了吧,這就是一種關(guān)系,自身對(duì)自身的關(guān)聯(lián)關(guān)系.這就有點(diǎn)類似樹的結(jié)構(gòu)了.下面用一個(gè)例子演示這種關(guān)系,代碼來(lái)源于孫MM的<<精通Hibernate>>一書.
    public class Category implements Serializable {

        private Long id;

        private String name;

        private Category parentCategory;

        private Set childCategories;

        public Category(String name, mypack.Category parentCategory, Set childCategories) {
            this.name = name;
            this.parentCategory = parentCategory;
            this.childCategories = childCategories;
        }

        public Category() {
        }

        public Category(Set childCategories) {
            this.childCategories = childCategories;
        }

        public Category getParentCategory() {
            return this.parentCategory;
        }

        public void setParentCategory(Category parentCategory) {
            this.parentCategory = parentCategory;
        }

        public Set getChildCategories() {
            return this.childCategories;
        }

        public void setChildCategories(Set childCategories) {
            this.childCategories = childCategories;
        }
       //為了節(jié)省空間省略了id,name屬性的訪問(wèn)方法

    }
    <hibernate-mapping >
      <class name="mypack.Category" table="CATEGORIES" >
        <id name="id" type="long" column="ID">
          <generator class="increment"/>
        </id>
        <property name="name" type="string" >
            <column name="NAME" length="15" />
        </property>
        <set
            name="childCategories"
            cascade="save-update"
            inverse="true"
            >
            <key column="CATEGORY_ID" />
            <one-to-many class="mypack.Category" />
         </set>  
       <many-to-one
            name="parentCategory"
            column="CATEGORY_ID"
            class="mypack.Category"
           />
      </class>
    </hibernate-mapping>
          我覺得這種方式其實(shí)和上邊的一對(duì)多,一對(duì)一關(guān)系一樣,只不過(guò)兩個(gè)用的都是同一個(gè)類罷了.看一下例子理解上應(yīng)該很簡(jiǎn)單.      

    posted on 2007-05-23 15:07 我心依舊 閱讀(5578) 評(píng)論(8)  編輯  收藏

    FeedBack:
    # re: 分析Hibernate映射的關(guān)聯(lián)關(guān)系
    2007-06-16 10:26 | qiudawei115
    不錯(cuò),你的blog已被收藏,希望多些點(diǎn)這樣的好文章  回復(fù)  更多評(píng)論
      
    # re: 分析Hibernate映射的關(guān)聯(lián)關(guān)系
    2007-07-29 20:07 | hanhong
    不錯(cuò)  回復(fù)  更多評(píng)論
      
    # re: 分析Hibernate映射的關(guān)聯(lián)關(guān)系
    2007-11-06 00:12 | 視頻資訊搜索-視搜網(wǎng)
    視頻資訊搜索-視搜網(wǎng)http://vsoso.cn  回復(fù)  更多評(píng)論
      
    # re: 分析Hibernate映射的關(guān)聯(lián)關(guān)系
    2007-12-20 15:00 | hermoine
    寫的很好,很易懂!繼續(xù)努力!  回復(fù)  更多評(píng)論
      
    # re: 分析Hibernate映射的關(guān)聯(lián)關(guān)系
    2008-02-20 10:30 | deity
    見解透徹,收藏了.  回復(fù)  更多評(píng)論
      
    # re: 分析Hibernate映射的關(guān)聯(lián)關(guān)系
    2008-03-06 18:33 | 吳丹
    寫的還真詳細(xì),很好啊  回復(fù)  更多評(píng)論
      
    # re: 分析Hibernate映射的關(guān)聯(lián)關(guān)系
    2008-09-02 13:52 | 小峰峰
    非常非常的不錯(cuò)!我找了很久了,我有個(gè)問(wèn)題想請(qǐng)教<many-to-one name="">里面一定是類的屬性名嗎?還有customer類中為什么用集合啊?謝謝!  回復(fù)  更多評(píng)論
      
    # re: 分析Hibernate映射的關(guān)聯(lián)關(guān)系
    2015-10-13 10:40 | mmocake
    一對(duì)多,一對(duì)一關(guān)系一樣  回復(fù)  更多評(píng)論
      

    只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。


    網(wǎng)站導(dǎo)航:
     
    主站蜘蛛池模板: 亚洲精品视频免费观看| 久久久99精品免费观看| 最近免费中文字幕大全视频| 亚洲国产精品免费在线观看| av无码久久久久不卡免费网站| 亚洲午夜久久久精品影院| 84pao强力永久免费高清| 亚洲欧洲春色校园另类小说| 67194国产精品免费观看| 亚洲精品日韩专区silk| 国产麻豆视频免费观看| 国内精品久久久久影院亚洲| 永久久久免费浮力影院| 深夜a级毛片免费无码| 亚洲色偷偷综合亚洲AVYP| 男人的天堂网免费网站| 91亚洲精品麻豆| 麻豆成人精品国产免费| 一个人看的免费视频www在线高清动漫 | 亚洲成人免费在线| 亚洲第一永久在线观看| 久久不见久久见中文字幕免费| 精品无码专区亚洲| 中文字幕精品无码亚洲字| 小草在线看片免费人成视久网| 亚洲理论精品午夜电影| 精品久久久久久久免费人妻| 日本视频免费观看| 亚洲视频在线观看地址| 性盈盈影院免费视频观看在线一区| 免费看一级高潮毛片| 亚洲av日韩av无码| 全免费一级午夜毛片| 九九热久久免费视频| 亚洲无成人网77777| 亚洲日本一区二区一本一道 | jizz中国免费| 亚洲国产精品美女| 亚洲人成色77777在线观看大 | 免费看搞黄视频网站| 亚洲www在线观看|