zoukankan      html  css  js  c++  java
  • NHibernate联合主键详细示例

    使用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(0, Name = "BN")]
    [KeyProperty(1, Name = "Id", Column = "Id", TypeType = typeof(string))]
    [KeyProperty(2, Name = "GroupNumber", Column = "GroupNumber", TypeType = typeof(string))]
    public virtual BaseInfo BN { get; set; }

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

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

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

    [Bag(0, Name = "Childs", Cascade = "all", Lazy = CollectionLazy.False, Inverse = true)]
    [Key(1)]
    [Column(2, Name = "ParentId")]
    [Column(3, Name = "GroupNumber")]
    [OneToMany(4, ClassType = typeof(Foo))]
    public virtual IList<Foo> Childs { get; set; }
    
    [ManyToOne(0, Name = "Parent", ClassType = typeof(Foo))] 
    [Column(1, Name = "ParentId")]
    [Column(2, 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(0, Name = "Parent", ClassType = typeof(Foo))] 
    [Column(1, Name = "ParentId")]
    [Column(2, Name = "ParentGroupNumber")]
    public virtual Foo Parent { get; set; }

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

    四、终于实现了,总结

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

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

    • 联合主键与联合外键字段不能重复,也不能共用
    • 注意重载的GetHashCode和Equals方法
    • GetHashCode返回实例的惟一标识
    • Equals判断是否相同实例的具体实现
  • 相关阅读:
    布尔值
    字典及字典的索引
    列表及列表的索引
    python之基本数据类型
    python之变量
    一个python程序运行的三大步骤
    编程语言分类与介绍
    应用程序的启动流程
    爬虫之PyQuery的base了解
    Django:web认识,jinja2模块,如何安装Django
  • 原文地址:https://www.cnblogs.com/cqhaibin/p/6263906.html
Copyright © 2011-2022 走看看