c# – 实体框架 – 外键未设置(0 / null)但导航属性不为null
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了c# – 实体框架 – 外键未设置(0 / null)但导航属性不为null,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含3582字,纯文字阅读大概需要6分钟。
内容图文
我已经坚持这个问题很长一段时间了.我正在使用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.对于OtherXmlDocId和OtherXmlDoc也是如此,除了Id为null之外.下面的添加方法:
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:所以在调试期间,当创建对象时,XmlDocId和OtherXmlDocId是0 / null,这是合乎逻辑的,因为它们尚未添加到它们各自的表中,因此没有Id.但是,根据我的理解,当我调用SaveChanges()时,EF会自动检测所有更改并应用它们,这应该更新FK(Source).
编辑II:解决问题的方法是在调用SaveChanges()之后显式设置FK(Template.XmlDocId = Template.XmlDoc.Id)但是当然这不是我所追求的解决方案.我使用EF来抽象这些手动操作并为我处理所有操作.
解决方法:
在您的一对一关系中,Template是主体,而XmlDoc和OtherXmlDoc是依赖者.这意味着Template没有外键,但相反 – XmlDoc / OtherXmlDoc中的PK Id列也是模板的FK.因此,XmlDocId和OtherXmlDocId列在关系中不起任何作用,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);
}
内容总结
以上是互联网集市为您收集整理的c# – 实体框架 – 外键未设置(0 / null)但导航属性不为null全部内容,希望文章能够帮你解决c# – 实体框架 – 外键未设置(0 / null)但导航属性不为null所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。