NHibernate Issues之1255:联合主键(composite-id)
2009-10-18 19:57 李永京 阅读(10285) 评论(15) 编辑 收藏 举报本节内容
概览
这个系列是以博客形式整理关于NHibernate的Issues。记录一些零碎的小例子,通过零零碎碎的整理,可以巩固自己的知识和扩展我们的知识面。这些小例子也可以适当的在项目中呈现。
这次看看联合主键(composite-id)。
实例
有时建模需要,需要把几个属性组合了一个标识符,就用到了联合主键(composite-id),我们使用它,就难免遇到了一些问题,比如如何映射,如何等同性比较,如何查询等操作。
1.Domain
这里定义一个Domain——CustomerValue,这个持久化类有两个属性:联合主键(composite-id)和Value。
我们使用联合主键,首先需要定义联合主键的类型,同时这个类必须重载Equals()和GetHashCode()方法,来实现组合的标识符等同性判断。 例如下面的CustomerValue持久化类,其联合主键类型为CustomerValueId,就是Customer持久化类与int类型的复合形式,在CustomerValueId类中重载了Equals()和GetHashCode()方法。
public class Customer { public virtual int Id { get; set; } public virtual string Name { get; set; } } public class CustomerValue { public virtual CustomerValueId Id { get; set; } public virtual decimal Value { get; set; } } public class CustomerValueId : IEquatable<CustomerValueId> { private int? _requestedHashCode; public Customer Customer { get; set; } public int CustomKey { get; set; } public bool Equals(CustomerValueId other) { if (ReferenceEquals(null, other)) { return false; } if (ReferenceEquals(this, other)) { return true; } return Equals(other.Customer, Customer) && other.CustomKey == CustomKey; } public override bool Equals(object obj) { if (ReferenceEquals(null, obj)) { return false; } if (ReferenceEquals(this, obj)) { return true; } if (obj.GetType() != typeof(CustomerValueId)) { return false; } return Equals((CustomerValueId)obj); } public override int GetHashCode() { if (!_requestedHashCode.HasValue) { unchecked { _requestedHashCode = ((Customer != null ? Customer.GetHashCode() : base.GetHashCode()) * 397) ^ CustomKey; } } return _requestedHashCode.Value; } }
2.Mapping
需要对两个Domain映射,这里注重理解下联合主键的映射,这里是把类的Customer属性和CustomerKey属性组合成为标识符属性。<key-many-to-one>节点用于映射子元素,一般都是Domain(多对一关系的);<key-property>节点用于映射属性。在这里我需要在<key-many-to-one>节点中设置not-found attribute来避免在运行时抛出异常,就是设置not-found="ignore"。
<class name="Customer"> <id name="Id"> <generator class="hilo" /> </id> <property name="Name" /> </class> <class name="CustomerValue" > <composite-id name="Id" class="CustomerValueId"> <key-many-to-one name="Customer" column="CustomerId" not-found="ignore" /> <key-property name="CustomKey" /> </composite-id> <property name="Value" /> </class>
3.Test
做完了上面的步骤,测试一下,保存一个CustomerValue。查询验证下并删除。这个测试很简单,就不贴出SQL结果了。
using (var session = OpenSession()) { using (var tx = session.BeginTransaction()) { var customer = new Customer { Name = "李永京" }; session.Save(customer); var customerValue = new CustomerValue { Id = new CustomerValueId { Customer = customer, CustomKey = 20012 }, Value = 1255.0m }; session.Save(customerValue); tx.Commit(); } using (var tx = session.BeginTransaction()) { var customerValue = session.CreateQuery("from CustomerValue c where c.Value=:value") .SetDecimal("value", 1255.0m) .UniqueResult<CustomerValue>(); session.Delete(customerValue); session.Delete("from Customer"); tx.Commit(); } }
参考资料
NHibernate Forge:NHibernate Reference Documentation(在线NHibernate参考文档5.1.5节composite-id介绍)
Adam Aldrich:Nhibernate Session.Get using a Composite Id(我们使用联合主键之后如何用ISession.Get<T>(object id)方法查询这个Domain呢?)