- NHibernate 教程
- NHibernate - 主页
- NHibernate - 概述
- NHibernate - 架构
- NHibernate-Orm
- NHibernate - 环境设置
- NHibernate - 入门
- NHibernate - 基本 Orm
- NHibernate - 基本 Crud 操作
- NHibernate - 分析器
- 将 Intelliesnse 添加到映射文件
- NHibernate - 数据类型映射
- NHibernate - 配置
- NHibernate - 覆盖配置
- NHibernate - 批量大小
- NHibernate - 缓存
- NHibernate - 映射组件
- NHibernate - 关系
- NHibernate - 集合映射
- NHibernate - 级联
- NHibernate - 延迟加载
- NHibernate - 逆关系
- NHibernate - 加载/获取
- NHibernate - Linq
- NHibernate - 查询语言
- NHibernate - 标准查询
- NHibernate - QueryOver 查询
- NHibernate - 原生 Sql
- NHibernate - 流畅的Hibernate
- NHibernate 有用资源
- NHibernate - 快速指南
- NHibernate - 有用的资源
- NHibernate - 讨论
NHibernate - 集合映射
在本章中,我们将介绍如何表示集合。我们可以在 NHibernate 中使用不同类型的集合,例如 -
- 列表
- 套
- 包
现在,从.NET的角度来看,我们通常处理列表或类似非常简单的数据结构、列表、字典。.NET 没有多种不同的集合类型。那么为什么 NHibernate 需要所有这些不同的类型呢?它真的回到了数据库。
列表
列表是元素的有序集合,这些元素不一定是唯一的。
我们可以使用IList <T>来映射它。
因此,尽管我们通常可能有一个地址列表,并且从应用程序的角度来看,我们知道元素是唯一的,但列表中没有任何内容可以阻止我们在该列表中插入重复元素。
放
集合是唯一元素的无序集合。如果您尝试将 2 个重复元素插入到集合中,则会抛出异常。
NHibernate 中对此没有任何具体说明。
这只是一个通用集合实现的便捷方法。如果您使用的是 .NET 4,则可以使用新的HashSet <T>来表示这些,但在大多数 NHibernate 应用程序中,我们表示这是一个 ISet。
它是无序的,如果您从数据库中提取地址列表或订单列表,您不知道它们的顺序是什么,除非您放入特定的 Order by 子句。
因此,一般来说,您从数据库中提取的数据是集合。
它们是无序元素的独特集合。
包
我们在数据库世界中看到的另一个常见集合是包,它就像一个集合,只是它可以包含重复的元素。
在 .NET 世界中,我们用 IList 来表示这一点。
集合可能是最常见的,但根据您的应用程序,您也会看到列表和包。让我们看一下上一章中的customer.hbm.xml文件,其中定义了 Set 订单。
<?xml version = "1.0" encoding = "utf-8" ?> <hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2" assembly = "NHibernateDemo" namespace = "NHibernateDemo"> <class name = "Customer"> <id name = "Id"> <generator class = "guid.comb"/> </id> <property name = "FirstName"/> <property name = "LastName"/> <property name = "AverageRating"/> <property name = "Points"/> <property name = "HasGoldStatus"/> <property name = "MemberSince" type = "UtcDateTime"/> <property name = "CreditRating" type = "CustomerCreditRatingType"/> <component name = "Address"> <property name = "Street"/> <property name = "City"/> <property name = "Province"/> <property name = "Country"/> </component> <set name = "Orders" table = "`Order`"> <key column = "CustomerId"/> <one-to-many class = "Order"/> </set> </class> </hibernate-mapping>
正如您所看到的,我们已将订单集合映射为一个集合。请记住,集合是唯一元素的无序集合。
现在,如果查看 Customer 类,您将看到 Orders 属性是使用 ISet 定义的,如以下程序所示。
public virtual ISet<Order> Orders { get; set; }
现在,当运行该应用程序时,您将看到以下输出。
New Customer: John Doe (00000000-0000-0000-0000-000000000000) Points: 100 HasGoldStatus: True MemberSince: 1/1/2012 12:00:00 AM (Unspecified) CreditRating: Good AverageRating: 42.42424242 Orders: Order Id: 00000000-0000-0000-0000-000000000000 Order Id: 00000000-0000-0000-0000-000000000000 Reloaded: John Doe (1f248133-b50a-4ad7-9915-a5b8017d0ff1) Points: 100 HasGoldStatus: True MemberSince: 1/1/2012 12:00:00 AM (Utc) CreditRating: Good AverageRating: 42.4242 Orders: Order Id: c41af8f2-7124-42a7-91c5-a5b8017d0ff6 Order Id: 657f6bb0-1f42-45fc-8fc7-a5b8017d0ff7 The orders were ordered by: John Doe (1f248133-b50a-4ad7-9915-a5b8017d0ff1) Points: 100 HasGoldStatus: True MemberSince: 1/1/2012 12:00:00 AM (Utc) CreditRating: Good AverageRating: 42.4242 Orders: Order Id: c41af8f2-7124-42a7-91c5-a5b8017d0ff6 Order Id: 657f6bb0-1f42-45fc-8fc7-a5b8017d0ff7 John Doe (1f248133-b50a-4ad7-9915-a5b8017d0ff1) Points: 100 HasGoldStatus: True MemberSince: 1/1/2012 12:00:00 AM (Utc) CreditRating: Good AverageRating: 42.4242 Orders: Order Id: c41af8f2-7124-42a7-91c5-a5b8017d0ff6 Order Id: 657f6bb0-1f42-45fc-8fc7-a5b8017d0ff7 Press <ENTER> to exit...
如果集合中的项目不需要是唯一的,如果您可以有多个具有相同主键的订单在此集合中多次出现,那么最好将其映射为包,如以下程序所示。
<bag name = "Orders" table = "`Order`"> <key column = "CustomerId"/> <one-to-many class = "Order"/> </bag>
现在,如果您运行此应用程序,您将得到一个异常,因为如果我们查看客户类,您会注意到订单在 C# 代码中被标记为 ISet。
因此,我们还需要将其更改为 IList,然后在这里,我们需要在构造函数中将 HashSet 更改为 List。
public class Customer { public Customer() { MemberSince = DateTime.UtcNow; Orders = new List<Order>(); } public virtual Guid Id { get; set; } public virtual string FirstName { get; set; } public virtual string LastName { get; set; } public virtual double AverageRating { get; set; } public virtual int Points { get; set; } public virtual bool HasGoldStatus { get; set; } public virtual DateTime MemberSince { get; set; } public virtual CustomerCreditRating CreditRating { get; set; } public virtual Location Address { get; set; } public virtual IList<Order> Orders { get; set; } public virtual void AddOrder(Order order) { Orders.Add(order); order.Customer = this; } public override string ToString() { var result = new StringBuilder(); result.AppendFormat("{1} {2} ({0})\r\n\tPoints: {3}\r\n\tHasGoldStatus: {4}\r\n\tMemberSince: {5} ({7})\r\n\tCreditRating: {6}\r\n\tAverageRating: {8}\r\n", Id, FirstName, LastName, Points, HasGoldStatus, MemberSince, CreditRating, MemberSince.Kind, AverageRating); result.AppendLine("\tOrders:"); foreach(var order in Orders) { result.AppendLine("\t\t" + order); } return result.ToString(); } }
当您运行该应用程序时,您将看到相同的Behave。但是,现在我们可以让一个订单在同一个集合中多次出现。
John Doe (00000000-0000-0000-0000-000000000000) Points: 100 HasGoldStatus: True MemberSince: 1/1/2012 12:00:00 AM (Unspecified) CreditRating: Good AverageRating: 42.42424242 Orders: Order Id: 00000000-0000-0000-0000-000000000000 Order Id: 00000000-0000-0000-0000-000000000000 Reloaded: John Doe (fbde48f5-d620-4d1c-9a7f-a5b8017c3280) Points: 100 HasGoldStatus: True MemberSince: 1/1/2012 12:00:00 AM (Utc) CreditRating: Good AverageRating: 42.4242 Orders: Order Id: 6dd7dbdb-354f-4c82-9c39-a5b8017c3286 Order Id: 9b3e2441-a81b-404d-9aed-a5b8017c3287 The orders were ordered by: John Doe (fbde48f5-d620-4d1c-9a7f-a5b8017c3280) Points: 100 HasGoldStatus: True MemberSince: 1/1/2012 12:00:00 AM (Utc) CreditRating: Good AverageRating: 42.4242 Orders: Order Id: 6dd7dbdb-354f-4c82-9c39-a5b8017c3286 Order Id: 9b3e2441-a81b-404d-9aed-a5b8017c3287 John Doe (fbde48f5-d620-4d1c-9a7f-a5b8017c3280) Points: 100 HasGoldStatus: True MemberSince: 1/1/2012 12:00:00 AM (Utc) CreditRating: Good AverageRating: 42.4242 Orders: Order Id: 6dd7dbdb-354f-4c82-9c39-a5b8017c3286 Order Id: 9b3e2441-a81b-404d-9aed-a5b8017c3287 Press <ENTER> to exit...