gpt4 book ai didi

c# - 将对象添加到实体集会创建不需要的新数据库行(多对多)。与现有对象建立关系?

转载 作者:太空宇宙 更新时间:2023-11-03 14:14:23 25 4
gpt4 key购买 nike

我是长期读者 - 第一次张贴 - 在对这个 Entity Framework 问题进行了无数小时的研究之后,我觉得我需要一些帮助。我是 C# 新手和编程新手,所以请多多包涵。

我的数据模型中有一个相当简单的多对多关系,带有 Web 2.0 样式的“标签”。类次有标签,用户有标签。

我正在从缓存中进行转换。当我尝试制作类次副本时,从缓存副本中复制类次详细信息和标签,我执行以下操作(简化)。

Data.Shift s = new Data.Shift();

/* copy over a bunch of stuff from the cached shift object. I'll spare the boring details */

foreach (var t in shift.Tags) { //shift object is a cached object

Data.Tag dt = new Data.Tag
{
TagID = t.TagID,
Name = t.Name,
OrgID = t.OrgID,
};
s.Tags.Add(dt);
}

即使我已经在新的 Data.Tag 对象中明确设置了 TagID,在 SaveChanges() 上一个新的标签被插入到数据库中,而不是仅仅在数据库中已经存在的标签上创建一个关系。 TagID是PK标识列

当我尝试以下代码时:

foreach (var t in shift.Tags){
s.Tags.Add(t)
}

它显然失败了,因为移位是从与当前请求的上下文不同的对象上下文中缓存的。

有什么想法吗?据我所知:

  • 出于对性能的考虑,清除缓存对我来说不是一个选择——我认识到如果我在相同的对象上下文中完成这项工作,整个问题就会消失。
  • 此外,出于性能方面的考虑,在当前上下文中从数据库中提取 Data.Tag 不是一种选择。

这似乎是我正在尝试做的一件非常简单的事情......

编辑好的——我已经尝试用这两种解决方案更新我的代码,但我都遇到了麻烦。让我们试试第一个:

// ctx is grabbed up here from HttpContext.Current.Items

Data.Shift s = new Data.Shift();

/* copy over a bunch of stuff from the cached shift object. I'll spare the boring details */

foreach (var t in shift.Tags) { //shift object is a cached object

Data.Tag dt = new Data.Tag
{
TagID = t.TagID,
Name = t.Name,
OrgID = t.OrgID,
};
ObjectStateEntry entry;
ctx.ObjectStateManager.TryGetObjectStateEntry(t.EntityKey, out entry);
if (entry == null || entry.State == EntityState.Detached) {
ctx.Tags.Attach(t);
}
s.Tags.Add(dt);
}

这是抛出以下错误:

  An entity object cannot be referenced by multiple instances of IEntityChangeTracker.

我猜我得到这个错误的原因是因为 shift 对象及其标签被从缓存中拉出(显然来自不同的上下文)。想法?

最佳答案

是否将 TAgId 设置为现有值并不重要。 EF 很愚蠢——它不会为你做任何更新插入/合并。它不会检查是否存在相同的实体,如果 TagId 是在数据库中自动生成的,它甚至会在您调用 SaveChanges 时丢弃您的值。

你必须做什么?您必须手动告知 EF Tag 不是新实体。

试试:

Data.Shift s = new Data.Shift();
context.Shifts.AddObject(s); // Add a new shift

foreach (var t in shift.Tags)
{ //shift object is a cached object
Data.Tag dt = new Data.Tag
{
TagID = t.TagID,
Name = t.Name,
OrgID = t.OrgID,
};
context.Tags.Attach(Tag); // Attach an existing tag
s.Tags.Add(dt);
}
context.SaveChanges();

Data.Shift s = new Data.Shift();
foreach (var t in shift.Tags)
{ //shift object is a cached object
Data.Tag dt = new Data.Tag
{
TagID = t.TagID,
Name = t.Name,
OrgID = t.OrgID,
};
s.Tags.Add(dt);
}

context.Shifts.AddObject(s);
// Now shift and all tags are added
// Change the state of each tag to unchanged
foreach (var tag in s.Tags)
{
context.ObjectStateManager.ChangeEntityState(tag, EntityState.Unchanged);
}
context.SaveChanges();

关于c# - 将对象添加到实体集会创建不需要的新数据库行(多对多)。与现有对象建立关系?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6933380/

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