我正在使用在Webapi请求上执行完所有操作后调用dbcontext.SaveChanges的工作单元模式。在请求的一部分中,我向dbcontext添加了一个新客户。
dbContext.Customers.Add(new Customer());
稍后在请求中(通常在域事件处理程序内部),我使用相同的dbcontext将客户撤回。
_dbContext.Customers.FirstOrDefault(x => x.Id == id);
public abstract class Customer
{
public Customer()
{
Id = Guid.NewGuid();
}
}
我已经验证了dbContext.Customers.Local有我期望的对象,但它似乎没有取出本地对象。这可能是因为Customer是一个抽象类,由DirectCustomer和InDirectCustomer实现?
为什么呢?我可以通过配置更改此行为吗?也许我必须合并本地和数据库结果(有点hacky)。
更新:
class Program
{
static void Main(string[] args)
{
MyDbContext context = new MyDbContext();
Guid customerGuid = Guid.NewGuid();
context.Customers.Add(new DirectCustomer()
{
Id = customerGuid
});
// This does not work, customerFromLocal1 is null
var customerFromLocal1 = context.Customers.FirstOrDefault(x => x.Id == customerGuid);
// This does work, customerFromLocal2 is NOT null
var customerFromLocal2 = context.Customers.Find(customerGuid);
}
}
public class MyDbContext : Microsoft.EntityFrameworkCore.DbContext
{
public DbSet<Customer> Customers { get; set; }
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSqlServer("server=.\\sqlexpress;integrated security=true;database=EFCoreDeepDive2");
base.OnConfiguring(optionsBuilder);
}
protected override void OnModelCreating(ModelBuilder builder)
{
builder.Entity<DirectCustomer>();
builder.Entity<IndirectCustomer>();
}
}
public abstract class Customer
{
public Guid Id { get; set; }
}
public class DirectCustomer : Customer
{
}
public class IndirectCustomer : Customer
{
public Guid ParentCustomerId { get; set; }
}
答案 0 :(得分:3)
在EF Core中,Linq运算符(如FirstOrDefault(),ToList()等&amp;它们在异步中的对应部分)会导致对服务器评估查询。对于查询,服务器数据是真实的来源。如果返回对象已经在内存中加载,它会组合返回对象,但它会首先检查服务器。
当您向上下文添加新的实体对象时,该对象存在于changetracker中,但在您调用SaveChanges()
之前它不会保存到服务器。所以,在添加实体&amp;在调用SaveChanges
之前,任何将针对服务器评估的查询都没有关于新添加的实体的信息,也不会返回与之相关的任何结果。
如果您尝试通过当前上下文实例中的键值找到实体对象(可能已保存或未保存),则使用context.DbSet.Find()
方法(还可以在上下文中定义其他变体。查找方法首先检查ChangeTracker以找到对象,其中包含所有对象加载的内存和添加/修改的对象。如果找不到,那么它将从服务器加载对象。在您的情况下,因为您要查找添加到的实体上下文但没有保存,Find
会给你预期的结果。
注意:DbSet.Local
包含当前上下文跟踪的DbSet类型的所有实体。因此,添加的实体在那里可用,但不能直接在DbSet上使用。 DbSet是IQueryable,以允许编写服务器查询。