指定的 LINQ 表达式包含对与不同上下文关联的查询的引用

Posted

技术标签:

【中文标题】指定的 LINQ 表达式包含对与不同上下文关联的查询的引用【英文标题】:The specified LINQ expression contains references to queries that are associated with different contexts 【发布时间】:2011-04-15 10:41:07 【问题描述】:

在此代码上出现此错误(这是一个 MVC 项目,我正在尝试将实体框架集成到该项目中):

        List<string> consultantSchoolList = new List<string>();

        // districts managed by consultant
        IQueryable<string> consultClients = rc.consultantDistrictsRepository.districtsForConsultant(userID);

        // schools managed by consultant
        if (consultClients != null)
        
            consultantSchoolList = (from c in rc.clientsRepository.Clients
                                    where (consultClients.Contains(c.cdsCode.Substring(0, 7)))
                                    select c.cdsCode).ToList();
        

在“consultantSchoolList =”行。

这是一个 MVC 项目,我正在使用存储在 HttpContext.Current 对象中的上下文对象。每个存储库都有一个存储上下文对象的私有变量,但每个变量都应该指向 HttpContext.Current Items 集合中的同一个对象。即使它们指向同一事物,这是否会被视为两种不同的上下文?

单步调试调试器中的代码显示,consultantDistrictsRepository 和 clientsRepository 这两个存储库的上下文对象确实指向 HttpContext.Current 对象中的同一个对象。

更新以下是我在每个存储库中定义上下文对象的方式:

    private SchedulerContext context;

    public EFConsultantDistricts()
    
        context = ContextHelper.GetContext();
    

而GetContext如下:

    public static SchedulerContext GetContext()
    
        if (!HttpContext.Current.Items.Contains("_db_context"))
        
            HttpContext.Current.Items.Add("_db_context", new SchedulerContext());
        
        return (SchedulerContext)HttpContext.Current.Items["_db_context"];
    

【问题讨论】:

clientRepository 和consultantDistrictsRepository 一样吗? 不,不是。它们是两个不同的存储库,具有两个不同的私有上下文变量。但是,两个上下文变量都指向 HttpContext.Current 对象中的同一个上下文对象。 我更新了我的问题以添加我用来初始化上下文对象的代码。我正在使用 Castle Windsor IoC 框架,因此每个存储库都会随每个请求一起处理。我假设上下文对象也会被处理掉。但无论如何,EF 抱怨的对象都在同一个请求中——实际上在同一个方法调用中。 【参考方案1】:

我发现了问题——我将经常请求的客户端列表缓存在客户端存储库的会话变量中:

            if (HttpContext.Current.Session["clientList"] == null)
            
                HttpContext.Current.Session["clientList"] = from c in context.Clients
                                                            where (c.Year == fiscalYear)
                                                            select c;
            
            return (IQueryable<Client>)HttpContext.Current.Session["clientList"];

由于 Session 对象在请求中持续存在,我猜它使用的是以前的上下文对象。我以为客户端列表会与上下文对象分开,但如果我使用的是 IQueryable,我想不会。

我讨厌每次都为此访问数据库,但我想我别无选择......至少在我把所有事情理顺之前是这样的。

【讨论】:

您不必每次都访问数据库。将信息缓存在非实体 POCO 而不是实体中,并根据需要将标量值传输到实体。 是的,其实我不知道我在想什么。这不是缓存客户端列表的内容,而是缓存查询,所以它有点毫无意义。 EF对我来说仍然有点神秘。这项技术的巨大学习曲线。

以上是关于指定的 LINQ 表达式包含对与不同上下文关联的查询的引用的主要内容,如果未能解决你的问题,请参考以下文章

指定的 LINQ 表达式包含对与不同上下文关联的查询的引用

指定的 LINQ 表达式包含对与不同上下文关联的查询的引用

错误:“指定的 LINQ 表达式包含对与不同上下文关联的查询的引用”[重复]

指定的 LINQ 表达式包含对与 c# 中的不同上下文错误关联的查询的引用

包含对与不同上下文关联的查询的引用的 LINQ 表达式

(Linq/Lambda) 使用 2 个 DBContext 连接 2 个或更多表