使用NHibernate实现一对多,多对一的关联很是简单,可如果要用复合主键实现确实让人有些淡淡的疼。虽然很淡疼但还是要去抹平这个坑,在下不才,愿意尝试。

以示例进入正文,源码下载地址

一、数据表关系图

很明显,他是一个自引用数表,实现无限级树结构的存储。

二、关键步骤

  • 注解如何实现复合主键

根据官方文档说明,联合主键最好是一个独立的类,需要重载Equals和GetHashCode方法,且标记为可序列化。代码如下:

[Serializable]
public class BaseInfo
{
public virtual string Id { get; set; }
public virtual string GroupNumber { get; set; } public override bool Equals(object obj)
{
var baseInfo = obj as BaseInfo;
if (baseInfo == null)
{
return false;
} return baseInfo.Id == this.Id && baseInfo.GroupNumber == this.GroupNumber;
}
public override int GetHashCode()
{
return base.GetHashCode();
}
}
  • 子类配置好联合主键
[CompositeId(, Name = "BN")]
[KeyProperty(, Name = "Id", Column = "Id", TypeType = typeof(string))]
[KeyProperty(, Name = "GroupNumber", Column = "GroupNumber", TypeType = typeof(string))]
public virtual BaseInfo BN { get; set; }

说明:
1.实现为引用BaseInfo类,而不是继承.

  • 实现一对 和 多对一的映射

这步没有多大难度,主要处理好注解的顺序即可,以及OneToMany时联合主键如何设置的问题.示例代码如下:

[Bag(, Name = "Childs", Cascade = "all", Lazy = CollectionLazy.False, Inverse = true)]
[Key()]
[Column(, Name = "ParentId")]
[Column(, Name = "GroupNumber")]
[OneToMany(, ClassType = typeof(Foo))]
public virtual IList<Foo> Childs { get; set; } [ManyToOne(, Name = "Parent", ClassType = typeof(Foo))]
[Column(, Name = "ParentId")]
[Column(, Name = "GroupNumber")]
public virtual Foo Parent { get; set; }

三、出错了,有Bug

  • childs没有数据

重载的GetHashCode方法有问题,返回值应该是联合主键HashCode,优化后的实现如下:

public override int GetHashCode()
{
return (this.Id + "|" + this.GroupNumber).GetHashCode(); //判断缓存是否存在,已此作为Key
}
  • 插入数据时报错,提示SqlParameterCollection的索引无效[索引溢出错误]

原因,最初在设计Parent的时候,与联合主键共用了一个字段GroupNumber,导致在NHibernate做映射转换的时候会多计算出一个需要填充的值,但SqlParameterCollection中又少一个位置。优化代码如下:

//外键与联合主键不要共用字段
[ManyToOne(, Name = "Parent", ClassType = typeof(Foo))]
[Column(, Name = "ParentId")]
[Column(, Name = "ParentGroupNumber")]
public virtual Foo Parent { get; set; }

说明:
1.由于联合外键与联合主键共用了一个字段,导致映射出错

四、终于实现了,总结

  • 类都必须可以序列化,也就是要还serializable标注
  • 继承BaseInfo实现联合主键(不推荐使用)

在Save时,如果用session.merge方法组合缓存与修改对象,返回值的主键会为Null

  • 联合主键与联合外键字段不能重复,也不能共用
  • 注意重载的GetHashCode和Equals方法
  • GetHashCode返回实例的惟一标识
  • Equals判断是否相同实例的具体实现

NHibernate联合主键详细示例的更多相关文章

  1. NHibernate composite-id联合主键配置

    NHibernate的联合主键配置比较复杂,初次配置可能需要花些时间,但只要我们理解了,掌握一定的步骤还是很容易的. 1.设计数据结构 Users:用户表 名称 Users 说明 用户表 序号 字段名 ...

  2. NHibernate 映射基础(第三篇) 简单映射、联合主键

    NHibernate 映射基础(第三篇) 简单映射.联合主键 NHibernate完全靠配置文件获取其所需的一切信息,其中映射文件,是其获取数据库与C#程序关系的所有信息来源. 一.简单映射 下面先来 ...

  3. Hibernate(5)—— 联合主键 、一对一关联关系映射(xml和注解) 和 领域驱动设计

    俗话说,自己写的代码,6个月后也是别人的代码……复习!复习!复习!涉及的知识点总结如下: One to One 映射关系 一对一单向外键(XML/Annotation) 一对一双向外键关联(XML/A ...

  4. SQL联合主键 查重

    2014年最后一天,今天在给数据库导入数据的时候,遇到一个问题,就是联合主键去重. 事情是这样的,现有一个表M,我想找个表中导入了许多数据,并需要将字段A(int)和B(int)联合设置为主键. 但是 ...

  5. Hibernate注解映射联合主键的三种主要方式

    今天在做项目的时候,一个中间表没有主键,所有在创建实体的时候也未加组件,结果报以下错误: org.springframework.beans.factory.BeanCreationException ...

  6. 联合主键用Hibernate注解映射的三种方式

    第一.将联合主键的字段单独放在一个类中,该类需要实现java.io.Serializable接口并重写equals和hascode,再将该类注解为@Embeddable,最后在主类中(该类不包含联合主 ...

  7. EntityFramework中Mapper怎么定义联合主键?

    HasKey(m => new { m.StoreId, m.CarTypeId, m.CarLevel}) 用“new {}”联合主键以“,”分隔形式定义

  8. SQL Server中的联合主键、聚集索引、非聚集索引、mysql 联合索引

    我们都知道在一个表中当需要2列以上才能确定记录的唯一性的时候,就需要用到联合主键,当建立联合主键以后,在查询数据的时候性能就会有很大的提升,不过并不是对联合主键的任何列单独查询的时候性能都会提升,但我 ...

  9. sql,联合主键,按id分组求版本号最大值的集合

    表结构如下: /* SQLyog v10.2 MySQL - 5.5.39 ************************************************************** ...

随机推荐

  1. UC编程:输入输出重定向(标准IO)

    [c] #include <stdio.h> #include <stdlib.h> int main() { FILE *fp; char szBuf[100]; /* 将屏 ...

  2. c#线程的几种启动方法

    一 启动普通线程 ThreadStart与ParameterizedThreadStart建立新线程 优缺点:简单,但难于管理,线程过多会影响系统的性能. 二 启动CLR线程池的工作者线程(普通线程和 ...

  3. 一步一步实现基于Task的Promise库(一)Promise的基本实现

    如果我们现在有一个需求,大概是先读取一个文件的内容,再把得到的内容传给后台去解析,最后把解析后的结果再保存到那个文件,按照最原始的做法代码就是下面这个样子的: //读取文件的原始内容 var read ...

  4. AngularJS学习笔记filter

    filter是对数据进行过滤操作,比如按某个字段搜索.格式化数据等等,是一个非常有用的接口.下面就简单介绍下它的用法. AngularJS自带的filter接口,|是filter的分隔符,参数用:分隔 ...

  5. status状态栏实现字符串走动

    <script type="text/javascript" language="javascript"> var i = 0; var str=& ...

  6. In C# 代码实现

    SOLID 设计原则 In C# 代码实现   [S] Single Responsibility Principle (单一职责原则) 认为一个对象应该仅只有一个单一的职责 namespace Si ...

  7. EntityFramework中支持BulkInsert扩展

    EntityFramework中支持BulkInsert扩展 本文为 Dennis Gao 原创技术文章,发表于博客园博客,未经作者本人允许禁止任何形式的转载. 前言 很显然,你应该不至于使用 Ent ...

  8. Windows Server 服务器安全配置

    Windows Server 服务器安全配置 好吧,我标题党了.我只了解一些基本的安全配置.如果你是大湿,请绕道或者给予我严厉的批评让我进步谢谢. 编辑这篇文章用的编辑器编辑的,当我单击查看的时候发现 ...

  9. iOS推送服务细节回顾

    iOS推送服务细节回顾 之前在做推送功能时候,就总结过一系列证书的制作,OC代码实现和服务器搭建等经验.又过了一段时间了,前前后后对推送服务做了多次的完善和优化,有iOS客户端的,还有本地服务器端的. ...

  10. s​s​h​配​置​公​钥​和​私​钥​登​陆​S​e​c​u​r​e​C​R​T

    在用windows时管理linux服务器时,常会用到SecureCRT.Xshell以及开源的putty.在我工作环境大多都是采用密码认证的方式进行登录.今天对学习了些SecureCRT的密钥登录方式 ...