gpt4 book ai didi

c# - Entity Framework - 未设置外键(0/空)但导航属性不为空

转载 作者:行者123 更新时间:2023-11-30 12:42:01 25 4
gpt4 key购买 nike

我已经被这个问题困扰了很长一段时间。我正在使用 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 不为 null,并且它具有 Id 3.OtherXmlDocIdOtherXmlDoc同理,只是Id为null,add方法示例如下:

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 会自动检测所有更改并应用它们,这应该会更新 FK(Source)。

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

最佳答案

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

很快,它们是多余的,您应该简单地删除它们并使用导航属性(在 SaveChanges 之后,非 null 导航属性的 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);
}

关于c# - Entity Framework - 未设置外键(0/空)但导航属性不为空,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35280807/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com