0

在我的应用程序中,我们DbContext为每个 HTTP 请求获取了一个新实例。在正常的工作流程中,我们创建一个实体并开始填充它的导航属性的一部分:

// Request 1
var foo = new Foo();
SessionStore.Add("Foo", foo);

// Request 2
var bar = BarDataService.GetBar(barId);

var foo = SessionStore.Get<Foo>("Foo");
foo.Bars.Add(bar);

// Request 3
var baz = BazDataService.GetBaz(bazId);

var foo = SessionStore.Get<Foo>("Foo");
foo.Baz = baz;

一旦对象图完全填充,我们将实体插入数据库:

var foo = SessionStore.Get<Foo>("Foo");
FooDataService.Add(foo); // BOOM!

Add 通常只是将实体添加到DbSet然后调用保存更改:

Set.Add(entity);
Context.SaveChanges();

显然,这失败了,因为该foo对象包含多个代理对象,每个代理对象都附加到不同的DbContexts. 我们决定的解决方案是分离我们收到的对象:

// Request 2 (modified)
var bar = BarDataService.GetBar(barId);
BarDataService.Detach(bar);
foo.Bars.Add(bar);

// Similar code for Request 3

var foo = SessionStore.Get<Foo>("Foo");
FooDataService.Add(foo); // Works

这行得通,除了它现在创建了BarBaz对象的新实例。

我在这里做错了什么?

4

1 回答 1

1

在添加到上下文之前,您必须附加相关的BarBaz对象。foo这会将实体置于状态Unchanged,EF 只会创建foo相关实体之间的关系,而不是在数据库中创建新的BarBaz对象:

foreach (var bar in foo.Bars)
    context.Bars.Attach(bar);
context.Bazs.Attach(foo.Baz);
context.Foos.Add(foo);
context.SaveChanges();
于 2013-03-02T00:42:11.757 回答