• [NHibernate]集合类(Collections)映射


    系列文章

    [Nhibernate]体系结构

    [NHibernate]ISessionFactory配置

    [NHibernate]持久化类(Persistent Classes)

    [NHibernate]O/R Mapping基础

    引言

    这部分不包括大量的.NET代码例子。我们假定你已经了解如何使用.NET自身的集合类框架(.NET's collections framework)和Set集合的概念。其实如果是这样,这里就真的没有什么东西需要学习了....用一句话来做个总结,你就用你已经掌握的知识来使用他们吧。

    持久化集合类(Persistent Collections)

    NHibernate可以持久化以下集合的实例,包括System.Collections.IDictionary,System.Collections.IList,lesi.Collectons.ISet和任何持久实体或值的数组。类型为System.Collections.IList的属性还可以使用“bag”语义来持久。

    警告:用于持久化的集合,除了集合接口外,不能保留任何实现这些接口的类所附加的语义(例如:lesi.Collectons.ListSet带来的迭代顺序iteration order)。所有的持久化集合,实际上都各自按照System.Collections.Hashtable,System.Collections.ArrayList,lesi.Collections.HashSet的语义直接工作。更深入地说,对于一个包含集合的属性来说,必须把.NET类型定义为接口(也就是IDictionary,IList或者ISet)。存在这个限制的原因是,在你不知道的时候,NHibernate暗中把你的IDictionary,IList和ISet的实例替换成了它自己的关于这些集合的实现。(所以在你的程序中,谨慎使用==操作符)

    1 Cat cat = new DomesticCat();
    2 Cat kitten = new DomesticCat();
    3 ...
    4 Iesi.Collections.ISet kittens = new Iesi.Collections.HashedSet(); 
    5 kittens.Add( kitten );
    6 cat.Kittens = kittens;
    7 session.Save( cat );
    8 kittens = cat.Kittens; // Okay, kittens collection is an ISet
    9 (Iesi.Collections.HashedSet)cat.Kittens; //Error! - a NHibernate.Collections.Set not Iesi.Collections.HashedSet

    集合遵从对值类型的通常规则:不能共享引用,与其包含的实体共存亡。由于存在底层的关联模型,集合不支持空值语义;

    并且NHibernate不会区分一个null的集合引用和一个不存在元素的空集合。

    集合类在被一个持久化对象引用的时候,会自动持久化,当不再被引用时将会自动删除。如果一个集合被从一个持久化对象传递到另一个,它的元素可能会从一个表转移到另一个表。你应该不需要对此特别关心。就如同你使用普通的 .NET集合类一样使用NHibernate的集合类,但是你需要确信使用前你理解了双向关联的语义(后面会讨论)。

    集合实例在数据库中根据指向对应实体的外键而得到区别。这个外键被称为集合的关键字。在NHibernate配置文件中使用 <key> 元素来映射这个集合的关键字。

    集合可以包含几乎所有的其他NHibernate类型, 包括所有的基本类型, 自定义类型,实体类型和组件。有一条重要的定义:在集合中的对象可以通过“传值”语义(完全依赖于集合自身)操作,也可以是一个指向其他实体的引用,拥有自己的生命周期。集合不能包含其他集合。这些被包含的元素的类型被称为集合元素类型。集合的元素在Hibernate中被映射为<element>, <composite-element>, <one-to-many>, <many-to-many> 或者<many-to-any>。前两种用传值语义操作元素,另外三种则映射实体关联。

    除了ISet和Bag之外的所有集合类型都有一个索引(index)字段,这个字段映射到一个数组或者IList的索引或者IDictionary的key。IDictionary的索引的类型可以是任何基本类型, 实体类型或者甚至是一个组合类型(但不能是一个集合类型)。数组和IList的索引肯定是整型(Int32)。在NHibernate配置文件中使用 <index>,<index-many-to-many> ,<composite-index> 或者<index-many-to-any>等元素来映射索引。

    集合类可以产生相当多种类的映射,涵盖了很多通常的关系模型。我们建议你练习使用schema生成工具, 以便对如何把不同的映射定义转换为数据库表有一个感性认识。

    映射集合(Mapping a Collection)

    值集合和多对多关联(Collections of Values and Many-To-Many Associations)
     一对多关联(One-To-Many Associations)

    <延迟初始化(延迟加载)(Lazy Initialization)

    集合排序(Sorted Collections)

    使用 <idbag>

    双向关联(Bidirectional Associations)

    双向关联允许通过关联的任一端访问另外一端。在NHibernate中,支持两种类型的双向关联:

    一对多(one-to-many):

    <set>或者<bag>值在一端, 单独值(非集合)在另外一端。

    多对多(many-to-many)

    两端都是<set>或<bag>值

    请注意NHibernate不支持带有索引的集合(IList,IDictionary或者数组)作为“多”的那一端的双向one-to-many关联,你必须使用集合或者bag映射。

    要建立一个双向的多对多关联,只需要映射两个many-to-many关联到同一个数据库表中,并再定义其中的一端为inverse(使用哪一端要根据你的选择)。这里有一个从一个类关联到他自身的many-to-many的双向关联的例子(每一个category都可以有很多items,每一个items可以属于很多categories):

     1 <class name name="NHibernate.Auction.Category, NHibernate.Auction">
     2  <id name="Id" column="ID"/>
     3  ...
     4  <bag name="Items" table="CATEGORY_ITEM" lazy="true">
     5   <key column="CATEGORY_ID" />
     6   <many-to-many class="NHibernate.Auction.Category, NHibernate.Auction" column="ITEM_ID" />
     7  </bag>
     8 </class>
     9 <class name="NHibernate.Auction.Item, NHibernate.Auction">
    10  <id name="Id" column="ID" />
    11  
    12  <!-- inverse end -->
    13  <bag name="Categories" table="CATEGORY_ITEM" inverse="true" lazy="true">
    14   <key column="ITEM_ID" />
    15   <many-to-many class="NHibernate.Auction.Category, NHibernate.Auction" column="CATEGORY_ID" />
    16  </bag>
    17 </class>

    如果只对关联的反向端进行了改变,这个改变不会被持久化。这表示NHibernate为每个双向关联在内存中存在两次表现,一个从A连接到B,另一个从B连接到A。如果你回想一下.NET对象模型,我们是如何在.NET中创建多对多关系的,这可以让你更容易理解:

    1 category.Items.Add( item ); // The category now "knows" about the relationship
    2 item.Categories.Add( category ); // The item now "knows" about the relationship
    3 session.Update( item ); // No effect, nothing will be saved!
    4 session.Update( category ); // The relationship will be saved

    非反向端用于把内存中的表示保存到数据库中,如果两端都进行了改变,我们会进行多余的INSERT/UPDATE,甚至可能得到外键冲突!这一点对双向的一对多关联也是一样的。
    要建立一个一对多的双向关联,你可以通过把一个一对多关联,作为一个多对一关联映射到同一张表的字段上,并且在“多”的那一端定义inverse=“true”,

     1 <class name="Eg.Parent, Eg>
     2  <id name="Id" column="id" />
     3  ...
     4  <set name="Children" inverse="true" lazy="true">
     5   <key column="parent_id" />
     6   <one-to-many class="Eg.Child, Eg" />
     7  </set>
     8 </class>
     9 <class name="Eg.Child, Eg">
    10  <id name="Id" column="id" />
    11  ....
    12  <many-to-one name="Parent" class="Eg.Parent, Eg" column="parent_id" />
    13 </class>

    在“一”这一端定义inverse=“true”不会影响级联操作,二者是不同的概念。

    三重关联(Ternary Associations)

    异类关联(Heterogeneous Associations)

    总结

    介绍对集合持久化,如何映射集合的方法。

    本文来自《NHibernate 中文文档》

     

  • 相关阅读:
    iPhone UIButton图标与文字间距设置【转】
    UIButton左边图片右边文字的做法
    UICollectionViewCell设置阴影
    ARC中__bridge, __bridge__transfer, __bridge_retained 关系
    NSFileHandle的用法(用于读写文件)
    自定义FrameWork
    IOS7 适配以及向下兼容问题
    IOS 7 UITableView cell lines不能靠左解决方法
    xml报错“cvc-complex-type.2.4.c: The matching wildcard is strict, but no declaration can be found for element”
    《Effective Java》笔记 :(一)创建和销毁对象
  • 原文地址:https://www.cnblogs.com/wolf-sun/p/3714577.html
Copyright © 2020-2023  润新知