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

Tru*_*uub 5 .net c# entity-framework

我已经被这个问题困扰了很长一段时间。我正在使用 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);
    }
}
Run Code Online (Sandbox Code Playgroud)

现在,当我在配置文件中植入数据库时​​,所有实体都已正确加载(FK 对应等)。但是,当我尝试如下添加一个对象时,XmlDocId是 0,而导航属性对象 (XmlDoc不为 null,它有3。除了Id 为 null 之外,情况Id也是如此。下面是 add 方法的示例:OtherXmlDocIdOtherXmlDoc

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();

}
Run Code Online (Sandbox Code Playgroud)

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

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

编辑 II:“解决”问题的方法是显式设置 FK 的 afterSaveChanges()被调用 ( Template.XmlDocId = Template.XmlDoc.Id) 但当然这不是我想要的解决方案。我使用 EF 抽象这些手动操作并为我处理这一切。

Iva*_*oev 2

在你们的关系one-to-oneTemplate主体XmlDocOtherXmlDoc是从属。这意味着没有外键,但相反 - /中的 PK 列也是 的 FK 。因此,列在关系中不起任何作用,并且 EF 将它们视为常规数据列,并且不将它们与导航属性同步。 TemplateIdXmlDocOtherXmlDocTemplateXmlDocIdOtherXmlDocId

很快,它们是多余的,您应该简单地删除它们并使用导航属性(在 后SaveChangesId非空导航属性的 始终与 相同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);
}
Run Code Online (Sandbox Code Playgroud)