DbContext不返回本地对象

时间:2017-09-12 20:37:20

标签: c# entity-framework-core

我正在使用在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; }
}

1 个答案:

答案 0 :(得分:3)

在EF Core中,Linq运算符(如FirstOrDefault(),ToList()等&amp;它们在异步中的对应部分)会导致对服务器评估查询。对于查询,服务器数据是真实的来源。如果返回对象已经在内存中加载,它会组合返回对象,但它会首先检查服务器。

当您向上下文添加新的实体对象时,该对象存在于changetracker中,但在您调用SaveChanges()之前它不会保存到服务器。所以,在添加实体&amp;在调用SaveChanges之前,任何将针对服务器评估的查询都没有关于新添加的实体的信息,也不会返回与之相关的任何结果。

如果您尝试通过当前上下文实例中的键值找到实体对象(可能已保存或未保存),则使用context.DbSet.Find()方法(还可以在上下文中定义其他变体。查找方法首先检查ChangeTracker以找到对象,其中包含所有对象加载的内存和添加/修改的对象。如果找不到,那么它将从服务器加载对象。在您的情况下,因为您要查找添加到的实体上下文但没有保存,Find会给你预期的结果。

注意:DbSet.Local包含当前上下文跟踪的DbSet类型的所有实体。因此,添加的实体在那里可用,但不能直接在DbSet上使用。 DbSet是IQueryable,以允许编写服务器查询。