鉴于以下代码,EF / DbContext如何了解对客户对象所做的更改:
class Program
{
static void Main()
{
using(var shopContext = new ShopContext())
{
var customer = shopContext.Customers.Find(7);
customer.City = "Marion";
customer.State = "Indiana";
shopContext.SaveChanges();
}
}
}
public class ShopContext : DbContext
{
public DbSet<Customer> Customers { get; set; }
}
public class Customer
{
public int Id { get; set; }
public string FirstName { get; set; }
public string LastName { get; set; }
public string City { get; set; }
public string State { get; set; }
}
谢谢
答案 0 :(得分:51)
当您从上下文加载实体时,它会保留一个额外的数据结构 - 让我们称之为条目。该条目包含两组值 - 原始值和当前值。当您执行SaveChanges
操作时,EF将通过您的客户实体并更新条目中的当前值,以便它们与您实体的实际状态相匹配 - 此操作称为检测更改。在SQL命令生成期间,EF将比较当前值和原始值,并构建SQL更新语句以修改数据库中的更改值。此操作称为快照更改跟踪 - EF会在条目中保留快照。
还有一种称为动态更改跟踪的替代方法,它会在您将值分配给实体的属性的同时修改条目中的当前值。动态更改跟踪具有特定要求(例如,实体中的所有属性必须为virtual
),因为它必须在运行时将您的类包装到动态代理中。这曾经是首选方式,但由于复杂方案中的某些性能问题,目前应将快照更改跟踪用作默认值。