使用跨多个数据上下文的POCO动态代理跟踪Entity Framework 4.0中的更改

时间:2010-04-15 01:17:31

标签: .net-4.0 poco entity-framework-4

我开始搞乱EF 4.0,因为我对POCO的可能性感到好奇......我想模拟断开连接的Web环境并编写以下代码来模拟这个:

  1. 将测试对象保存在数据库中。
  2. 检索测试对象
  3. 处理与我用来检索它的测试对象相关联的DataContext
  4. 更新测试对象
  5. 创建一个新的数据上下文,并在测试对象上保留对我的POCO对象生成的DynamicProxy中自动跟踪的更改。
  6. 问题是当我在上面的Test方法中调用dataContext.SaveChanges时,不会应用更新。当我检查其EntityStateTracker时,testStore实体显示状态为“已修改”,但当我在新的dataContext的Stores属性中查看它时,它不再被修改。我原本以为在new dataContext上调用Attach方法也会使对象的“Modified”状态结束,但事实并非如此。有什么我想念的吗?我肯定使用DynamicProxies进行自我跟踪POCO。

    private static void SaveTestStore(string storeName = "TestStore")
    {
      using (var context = new DataContext())
      {
        Store newStore = context.Stores.CreateObject();
        newStore.Name = storeName;
        context.Stores.AddObject(newStore);
        context.SaveChanges();
      }
    }
    
    private static Store GetStore(string storeName = "TestStore")
    {
      using (var context = new DataContext())
      {
        return (from store in context.Stores
                where store.Name == storeName
                select store).SingleOrDefault();
      }
    }
    
    [Test]
    public void Test_Store_Update_Using_Different_DataContext()
    {
      SaveTestStore();
      Store testStore = GetStore();
      testStore.Name = "Updated";      
    
      using (var dataContext = new DataContext())
      {
        dataContext.Stores.Attach(testStore);
        dataContext.SaveChanges(SaveOptions.DetectChangesBeforeSave);        
      }
    
      Store updatedStore = GetStore("Updated");
      Assert.IsNotNull(updatedStore);
    }
    

4 个答案:

答案 0 :(得分:8)

正如您稍后所述,您使用的是POCO生成器,而不是自跟踪实体生成器。

我也试过了,变得非常困惑。似乎代理类没有按预期工作,并且可能存在错误。然后又来了。 MSDN上没有一个例子尝试这样的东西,当他们在应用程序的不同层引用更新时(我们在这里做的事情),他们使用自我跟踪实体,而不是POCO代理。

我不确定这些代理是如何工作的,但它们确实存在某种状态(我设法在私有属性中找到“已修改”状态)。但似乎这个属性完全被忽略了。将属性附加到上下文时,上下文会向ObjectStateManager添加一个条目,并在其中存储更多状态更新。此时,如果您进行更改 - 它将被注册并应用。

问题在于,当你.Attach一个实体时 - 来自代理的Modified状态不会转移到上下文中的状态管理器。此外,如果您使用context.Refresh(),更新将被覆盖,并被遗忘!即使你将RefreshMode.ClientWins传递给它。我尝试将对象状态的state属性设置为modified,但无论如何都被覆盖了,原始设置被恢复了。

似乎EF中没有错误,唯一的方法就是使用这样的东西:

using (var db = new Entities())
{
    var newUser = (from u in db.Users
                    where u.Id == user.Id
                    select u).SingleOrDefault();
    db.Users.ApplyCurrentValues(user);
    db.SaveChanges();
}

这里还有一件事

Entitity Framework: Change tracking in SOA with POCO approach

似乎POCO不支持您正在寻找的方法,并且正如我所预期的那样,自我跟踪实体的创建是为了解决您正在测试的情况,而POCO的代理仅在他们创建的上下文中跟踪更改..或者看起来似乎......

答案 1 :(得分:5)

尝试

        db.ObjectStateManager.ChangeObjectState(user, System.Data.EntityState.Modified);

在致电SaveChanges之前

答案 2 :(得分:2)

在玩自我跟踪实体后,我意识到你的错误是什么。 您应该反而指示您希望数据上下文将您对其所做的新更改应用于数据库,而不是尝试将实体附加到数据上下文。

在这种情况下,请将“保存”代码更改为:

using (var dataContext = new DataContext())
{
    dataContext.Stores.ApplyChanges(testStore);
    dataContext.SaveChanges();        
}

至少我已经在我的本地机器上测试了它,并且在这次更新后它起作用了:) 希望这有帮助!

答案 3 :(得分:0)

我认为问题的根源是您对Context对象的管理。

使用POCO处理上下文时,不会通知该上下文中的实体它们不再与上下文关联。使用POCO进行的更改跟踪全部由上下文管理,因此您会遇到一些有趣的问题,其中POCO将表现为仍然附加到上下文但实际上它不是并且重新附加到另一个上下文应该引发关于附加的错误多个情境。

有一篇关于此的小帖子,你可能想在这里阅读: http://social.msdn.microsoft.com/forums/en-US/adodotnetentityframework/thread/5ee5db93-f8f3-44ef-8615-5002949bea71/

如果您切换到自我跟踪,我认为您会发现您的实体以您想要的方式运作。

另一个选项是将属性添加到poco的部分类中,以便在从用于加载它的上下文中分离POCO后手动跟踪更改。