在我的应用程序中,我们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
这行得通,除了它现在创建了Bar
和Baz
对象的新实例。
我在这里做错了什么?