Fluent NHibernate:如何创建一对多的双向映射?

Posted

技术标签:

【中文标题】Fluent NHibernate:如何创建一对多的双向映射?【英文标题】:Fluent NHibernate: How to create one-to-many bidirectional mapping? 【发布时间】:2010-09-23 13:31:17 【问题描述】:

基本问题:如何在 Fluent NHibernate 中创建双向一对多映射?

详情:

我有一个有很多孩子的父对象。在我的情况下,孩子没有父母是没有意义的,所以在数据库中,我希望父母的外键具有 NOT NULL 约束。我正在从 Fluent NHibernate 映射自动生成我的数据库。

我的父母有很多这样的子对象:

public class Summary

   public int id get; protected set;

   public IList<Detail> Details get; protected set;


public  class Detail

   public int id get; protected set;

   public string ItemName get; set;

  /* public Summary Owner get; protected set; */ //I think this might be needed for bidirectional mapping?

这是我开始时的映射:

public class SummaryMap : ClassMap<Summary>

    public SummaryMap()
    
        Id(x => x.ID);

        HasMany<Detail>(x => x.Details);
    


public class DetailMap : ClassMap<Detail>

    public DetailMap()
    
        Id(x => x.ID);

        Map(x => x.ItemName).CanNotBeNull();
    

在 Detail 表中,Summary_id 应该为 Not Null,因为在我的 如果 Detail 对象未附加到 摘要对象。但是,仅使用 HasMany() 映射会使 Summary_id 外键为空。

我在 NHibernate 文档 (http://www.hibernate.org/hib_docs/nhibernate/html/collections.html) 中发现“如果需要父级,请使用双向一对多关联”。

那么如何在 Fluent NHibernate 中创建双向一对多映射?

【问题讨论】:

【参考方案1】:

要获得与 Details 表中非空外键列的双向关联,您可以在 DetailsMap 类中添加建议的 Owner 属性、References(...).CanNotBeNull() 映射,并使摘要结束逆。

为避免两个关联方向有两个不同的外键列,您可以手动指定列名或以为两个方向提供相同列名的方式命名属性。在这种情况下,我建议您将 Details.Owner 属性重命名为 Details.Summary。

我制作了由增量生成的摘要 id 以避免插入表时出现问题,因为摘要当前除了 id 之外没有其他列。

域:

public class Detail

    public int id  get; protected set; 
    public string ItemName  get; set; 

    // Renamed to use same column name as specified in the mapping of Summary.Details
    public Summary Summary get; set; 


public class Summary

    public Summary()
    
        Details = new List<Detail>();
    

    public int id  get; protected set; 
    public IList<Detail> Details  get; protected set; 

映射:

public class DetailMap : ClassMap<Detail>

    public DetailMap()
    
        Id(x => x.id)
            .GeneratedBy.Native();

        Map(x => x.ItemName)
            .CanNotBeNull();

        References<Summary>(x => x.Summary)
            // If you don't want to rename the property in Summary,
            // you can do this instead:
            // .TheColumnNameIs("Summary_id")
            .CanNotBeNull();
    


public class SummaryMap : ClassMap<Summary>

    public SummaryMap()
    
        Id(x => x.id)
            .GeneratedBy.Increment();

        HasMany<Detail>(x => x.Details)
            .IsInverse()
            .AsBag(); // Use bag instead of list to avoid index updating issues
    

【讨论】:

真的没有办法让外键列“不可为空”而不在子上添加对父级的引用吗?

以上是关于Fluent NHibernate:如何创建一对多的双向映射?的主要内容,如果未能解决你的问题,请参考以下文章

Fluent Nhibernate Mapping - 值对象内的一对多?

如何在 Fluent NHibernate 中将一对一关系映射为复合键的一部分

Fluent NHibernate:约定/KeyColumn

Fluent NHibernate:如何在关系表上映射具有附加属性的多对多关系?

Fluent NHibernate Where 子句

使用 Fluent NHibernate 映射或模仿多对任意关系的选项?