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

时间:2016-02-08 22:45:30

标签: c# .net 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);
    }
}

现在,当我在配置文件中为数据库播种时,所有实体都被正确加载(FK对应等)。但是,当我尝试添加如下对象时,XmlDocId为0,而导航属性对象(XmlDoc不为空且它具有Id 3。{同样如此{1}}和OtherXmlDocId除了Id为空。以下添加方法的示例:

OtherXmlDoc

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

编辑我:所以在调试期间,当创建对象时,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(); } XmlDocId为0 / null这是合乎逻辑的,因为它们没有添加到各自的表还没有ID。但是,根据我的理解,当我致电OtherXmlDocId时,EF会自动检测所有更改并应用它们,这应该会更新FK(Source)。

编辑II :问题是“解决”问题是在调用SaveChanges()后明确设置FK(SaveChanges())但是当然这不是我追求的解决方案。我使用EF来抽象这些手动操作并为我处理它。

2 个答案:

答案 0 :(得分:1)

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

很快,它们是多余的,您应该只删除它们并使用导航属性(在SaveChanges之后,非空导航属性的Id将始终与{{1}相同})。添加反向导航属性也很不错,因此最终的模型/设置可能就像这样

entity.Id

答案 1 :(得分:0)

看看提及外键是否明确有帮助:

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; }
}