实体框架 - 未设置外键 (0 / null),但导航属性不为 null

本文关键字:null 导航 属性 框架 设置 实体 | 更新日期: 2023-09-27 18:33:57

我已经在这个问题上停留了很长时间。我正在使用 Fluent API Code-First 来设计一个数据库 (EF 6.1(。问题是,当我添加一个新对象时,我可以通过其导航属性访问该对象中的实体,但 FK 为 0 或 NULL(分别是必需和可选(。它们是一对一的关系,我尝试过双向和单向。

下面是代码的一部分(简化,但仅包含较少的实体(:

public class Template
{
    public int Id { get; set; }
    public int XmlDocId{ get; set; }
    public virtual XmlDoc XmlDoc { get; set; }
    public int? OtherXmlDocId{ get; set; }
    public virtual OtherXmlDoc OtherXmlDoc { get; set; }
}
public class XmlDoc 
{
    public int Id { get; set; }
    [Required]
    public string RawXml { get; set; }
}
public class OtherXmlDoc 
{
    public int Id { get; set; }
    [Required]
    public string RawXml { get; set; }
}
public class MyDbContext
{
    public virtual DbSet<Template> Templates { get; set; }
    public virtual DbSet<XmlDoc> XmlDocs { get; set; }
    public virtual DbSet<OtherXmlDoc> OtherXmlDocs { get; set; }

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        base.OnModelCreating(modelBuilder);
        ConfigureTemplates(modelBuilder);
    }
    private void ConfigureTemplates(DbModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Template>()
            .HasRequired<XmlDoc>(c => c.XmlDoc)
            .WithRequiredPrincipal()
            .WillCascadeOnDelete(true);
        // XML SCHEMA, delete with TT (optional)
        modelBuilder.Entity<Template>()
            .HasOptional<OtherXmlDoc>(c => c.OtherXmlDoc)
            .WithOptionalPrincipal()
            .WillCascadeOnDelete(true);
    }
}

现在,当我在配置文件中为数据库设定种子时,所有实体都已正确加载(FK 的对应等(。但是,当我尝试添加如下所示的对象时,XmlDocId为 0,而导航属性对象 ( XmlDoc 不为空,它Id 3。对于OtherXmlDocIdOtherXmlDoc也是如此,只是 Id 为空。下面是添加方法的示例:

public void Add(string xmlString1, string xmlString2)
{
    XmlDoc firstDoc = new XmlDoc { rawXml = xmlString1 };
    OtherXmlDoc secondDoc = new OtherXmlDoc { rawXml = xmlString2 };
    var entity = new Template
    {
        XmlDoc = firstDoc,
        OtherXmlDoc = secondDoc
    }
    context.Templates.Add(entity);
    context.SaveChanges();
}

我真的被困在了这个上面。我尝试更改很多东西,扁平化数据库,仅使用数据注释并浪费了很多时间。希望有人可以帮助:)。提前非常感谢!

编辑 I:因此,在调试期间,当创建对象时,XmlDocIdOtherXmlDocId为 0/null,这是合乎逻辑的,因为它们尚未添加到各自的表中,因此没有 Id。但是,据我了解,当我调用SaveChanges() EF时,EF会自动检测所有更改并应用它们,这应该会更新FK(源(。

编辑二:"解决"问题的是在SaveChanges()被称为(Template.XmlDocId = Template.XmlDoc.Id(后明确设置FK,但当然这不是我想要的解决方案。我使用 EF 抽象这些手动操作并为我处理所有这些操作。

实体框架 - 未设置外键 (0 / null),但导航属性不为 null

在你的one-to-one关系中,Template主体,而XmlDocOtherXmlDoc依赖者。这意味着Template没有外键,而是相反 - XmlDoc/OtherXmlDoc中的PK Id列也是FK Template。因此,XmlDocIdOtherXmlDocId列在关系中不发挥任何作用,EF 将它们视为常规数据列,并且不会将它们与导航属性同步。

简而言之,它们是多余的,您应该简单地删除它们并使用导航属性(SaveChanges之后,非空导航属性的Id将始终与entity.Id相同(。此外,最好添加反向导航属性,因此最终模型/设置可能是这样的

public class Template
{
    public int Id { get; set; }
    public virtual XmlDoc XmlDoc { get; set; }
    public virtual OtherXmlDoc OtherXmlDoc { get; set; }
}
public class XmlDoc
{
    public int Id { get; set; }
    [Required]
    public string RawXml { get; set; }
    public virtual Template Template { get; set; }
}
public class OtherXmlDoc
{
    public int Id { get; set; }
    [Required]
    public string RawXml { get; set; }
    public virtual Template Template { get; set; }
}
private void ConfigureTemplates(DbModelBuilder modelBuilder)
{
    modelBuilder.Entity<Template>()
        .HasRequired<XmlDoc>(t => t.XmlDoc)
        .WithRequiredPrincipal(d => d.Template)
        .WillCascadeOnDelete(true);
    modelBuilder.Entity<Template>()
        .HasOptional<OtherXmlDoc>(t => t.OtherXmlDoc)
        .WithOptionalPrincipal(d => d.Template)
        .WillCascadeOnDelete(true);
}

查看显式提及外键是否有帮助:

public class Template
{
public int Id { get; set; }
public int XmlDocId{ get; set; }
[ForeignKey("XmlDocId")]
public virtual XmlDoc XmlDoc { get; set; }
public int? OtherXmlDocId{ get; set; }
[ForeignKey("OtherXmlDocId")]
public virtual OtherXmlDoc OtherXmlDoc { get; set; }
}