NHibernate學習筆記(三):many-to-one/one-to-many/many-to-many關係映射

本文的內容:
  介紹NH如何處理對象間many-to-one,one-to-many和many-to-many的雙向映射關係(本文主要介紹了映射文件,對於類的生成可以跟據前兩篇描述的方法生成);

經驗教訓:

相關類圖:(下面的類圖包含一對一,多對一,一對多和多對多)img3.JPG
many-to-one:描述多對一的一種數據模型,它指定many一方是不能獨立存在的,我個人認爲many-to-one是NHB中保證數據有效性的最有用的一種映射,通過使用many-to-one能有效的防治孤兒記錄被寫入到數據表中。在本文描述的例子中,Student類和Classes類之間是多對一關係.

  通過many-to-one元素,可以定義一種常見的與另一個持久化類的關聯。這種關係模型是多對一關聯。(實際上是一個對象引用。)在映射文件中用many-to-one標籤描述此種關係.
None.gif<many-to-one
None.gif        
name="propertyName"(1)
None.gif        column
="column_name"(2)
None.gif        class
="ClassName"(3)
None.gif        cascade
="all|none|save-update|delete"(4)
None.gif        outer-join
="true|false|auto"(5)
None.gif        update
="true|false"(6)
None.gif        insert
="true|false"(7)
None.gif        property-ref
="propertyNameFromAssociatedClass" (8)
None.gif        access
="field|property|ClassName"(9)
None.gif        unique
="true|false" (10)
None.gif
/>

1.name:屬性名。指出many一方的類用哪個屬性和one一方的類關聯.
2.column:字段名(可選).指出many一方的類對應的數據表用哪個列和one一方的類對應的數據表關聯(兩表之間存在外鍵關聯);
3.class:關聯的類的名字(可選 - 默認是通過反射得到屬性類型);
4.cascade:指明哪些操作會從父對象級聯到關聯的對象(可選).cascade屬性允許下列值:: all, save-update, delete, none. 設置除了none以外的其它值會傳播特定的操作到關聯的(子)對象中。
5.outer-join:當設置hibernate.use_outer_join的時候,對這個關聯允許外連接抓取(可選 - 默認爲 auto).outer-join參數允許下列三個不同值: auto(使用外連接抓取關聯(對象),如果被關聯的對象沒有代理(proxy) ),true(一直使用外連接來抓取關聯),false(永遠不使用外連接來抓取關聯);
6.update,insert:指定對應的字段是否在用於UPDATE 和/或 INSERT的SQL語句中包含。如果二者都是false,則這是一個純粹的“外源性(derived)”關聯,它的值是通過映射到同一個(或多個)字段的某些其他屬性得到的,或者通過trigger(除法器),或者是其他程序(可選 - 默認爲 true
7.property-ref:指定關聯類的一個屬性,這個屬性將會和本外鍵相對應。如果沒有指定,會使用對方關聯類的主鍵(可選).property-ref屬性只應該用來對付老舊的數據庫系統,可能出現外鍵指向對方關聯表的是個非主鍵字段(但是應該是一個惟一關鍵字)的情況。這是一種十分醜陋的關係模型。比如說,假設Product類有一個惟一的序列號,它並不是主鍵;
8.access:NHibernate 用來訪問屬性的策略(可選 - 默認爲property
9.unique:允許產生外鍵列唯一約束的數據庫定義語言(DDL)(可選)

那麼關於Student的映射可能是:

None.gif<many-to-one name="Classes" column="cID" unique ="true"/>

只要在原Student.hbm.xml映射文件中添加many-to-one標籤就可以了.

對於Sturent類得添加一個屬性Classes:Classes

one-to-many:
一對多也是一種常見的數據模型,在按範式設計的數據庫中隨處可見。在NHB中通過one-to-many可以非常方便的處理這種模型,同時NHB還提供了級聯更新和刪除的功能,以保證數據完整性。在本文描述的例子中,Classes類和Student類是一對多的關係.

Classes類的映射文件:Classes.hbm.xml

None.gif<?xml version="1.0" encoding="utf-8" ?>
None.gif
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.0">
None.gif  
<class name="NHibernateTest.Classes,NHibernateTest" table="Classes">
None.gif    
<id name="ClassesID" column="ID" type="Int32" unsaved-value="0">
None.gif      
<generator class="identity"/>
None.gif    
</id>
None.gif    
<property name="ClassesName" column="ClassName" type="String" length="50"/>
None.gif
None.gif    
<bag name="StudentList" cascade="all"  inverse="true">
None.gif      
<key column="ID" />
None.gif      
<one-to-many class="NHibernateTest.Student,NHibernateTest" />
None.gif    
</bag>
None.gif  
</class>
None.gif
</hibernate-mapping>

  如映射文件所示,one-to-many標籤必須包含在標籤bag中(個人理解:多嘛,就用一個包裝起來~)
        bag標籤的name屬性指出Address對象用哪個屬性和Student對象關聯,inverse屬性使collection不更新連接(總之,這個屬性提高了性能,具體的情況請參考NHibernate的幫助文檔).
  key標籤的column屬性指出了Address對象對應的數據表用哪個字段和Student對象對應的數據表關聯
  one-to-many標籤的class屬性指出了Address和哪個對象關聯.

對於Classes類得創建一個StudentList : Student的屬性,用來描述與Student對象的關係.

many-to-many:多對多在數據庫中也是常見的數據模型,像用戶與組,用戶與權限等。多對多關係需要通過一箇中間表實現,element的就是讀取這個中間表中某列的值。在本文的例子中,Student類和Subject類是多對多的關係.

  Student的映射可能是:

None.gif    <bag name="SubjectList" table="r_Student_Subject" inverse="true" cascade="save-update" lazy="false">
None.gif      
<key column="StudentID" />
None.gif      
<many-to-many class="NHibernateTest.Subject,NHibernateTest" column="ID" outer-join="auto" />
None.gif    
</bag>

  在bag標籤中,加入了一個table屬性,它指定一個實現多對多的中間表

  完整的Student.hbm.xml源碼如下:

None.gif<?xml version="1.0" encoding="utf-8" ?>
None.gif
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.0">
None.gif  
<class name="NHibernateTest.Student,NHibernateTest" table="Users">
None.gif    
<id name="UserID" column="ID" type="Int32" unsaved-value="0">
None.gif      
<generator class="identity"/>
None.gif    
</id>
None.gif    
<property name="UserName" column="UserName" type="String" length="20"/>
None.gif    
<property name="Password" column="Password" type="String" length="20"/>
None.gif
None.gif    
<!--Student類和NativePlace類是一對一的關係-->
None.gif    
<one-to-one name="NativePlace" class="NHibernateTest.NativePlace,NHibernateTest" cascade="all" />
None.gif
None.gif    
<!--Student類和Classes類是多對一的關係-->
None.gif    
<many-to-one name="Classes" column="cID" unique ="true"/>
None.gif
None.gif    
<!--Student類和Address類是一對多的關係-->
None.gif    
<bag name="AddressList" cascade="all"  inverse="true">
None.gif      
<key column="ID" />
None.gif      
<one-to-many class="NHibernateTest.Address,NHibernateTest" />
None.gif    
</bag>
None.gif
None.gif    
<!--Student類和Subject類是多對多的關係-->
None.gif    
<bag name="SubjectList" table="r_Student_Subject" inverse="true" cascade="save-update" lazy="false">
None.gif      
<key column="StudentID" />
None.gif      
<many-to-many class="NHibernateTest.Subject,NHibernateTest" column="ID" outer-join="auto" />
None.gif    
</bag>
None.gif  
</class>
None.gif
</hibernate-mapping>
None.gif

  Student類得添加類型爲IList的屬性SubjectList表示與類Subject的關係.

  由於Student的映射關係比較複雜,本文就再熬述它的CRUD操作,具體的操作方法請參考本文附帶的代碼.

發佈了70 篇原創文章 · 獲贊 5 · 訪問量 31萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章