Entity framework EF Core 2.2 LINQ查询在EF Core 3.0中不工作

Entity framework EF Core 2.2 LINQ查询在EF Core 3.0中不工作,entity-framework,ef-core-3.0,Entity Framework,Ef Core 3.0,以下代码在EF Core 2.2上运行良好,但不在EF Core 3.0上运行 var items = (from asset in Context.Assets join assetCategory in Context.AssetCategories on asset.CategoryId equals assetCategory.Id group assetCategory by assetCategory.Id into asse

以下代码在EF Core 2.2上运行良好,但不在EF Core 3.0上运行

 var items = (from asset in Context.Assets
              join assetCategory in Context.AssetCategories on asset.CategoryId equals assetCategory.Id
              group assetCategory by assetCategory.Id into assetCategories
              select new AssetCategorySummary
              {
                  CategoryId = assetCategories.Key,
                  CategoryName = assetCategories.Select(p => p.CategoryName).FirstOrDefault(),
                  TotalAsset = assetCategories.Count()
              }).ToListAsync();
我得到的错误是:

有关详细信息,请处理LINQ表达式“AsQueryableSelect”

需要帮助请

这是由于EF Core 3.0中的一个问题,即:


因此,请以这样的方式编写查询:EF Core可以将表达式转换为T-SQL或将数据提取到内存中,然后进行查询。

原始查询有问题,但EF Core将其隐藏在地毯下,减慢了所有操作

当客户端评估在LINQtoSQL中引入并在实体框架中删除时,它是邪恶的。我不明白为什么人们认为把它添加回EF Core是个好主意,但现在它已经消失了,这是件好事。原始查询也不会在EF 6.2中运行

原始查询需要一些修复,这可能会导致性能改进。首先,ORM的工作是从关系和导航属性生成连接

其次,即使在SQL中,也不可能在SELECT子句中添加不属于GROUP BY或聚合的字段。除非使用窗口函数,否则没有与FirstOrDefault等效的聚合函数

要在SQL中获取类别名称,我们必须将其包含在GROUPBY中,或者使用CTE/子查询按ID分组,然后查找类别名称,例如:

SELECT CategoryID,CategoryName,Count(*)
FROM Assets inner join AssetCategories on CategoryID=AssetCategories.ID
GROUP BY CategoryID,CategoryName

LINQ中第一个查询的等价物是:

 var items = (from asset in Context.Assets
              join assetCategory in Context.AssetCategories on asset.CategoryId equals assetCategory.Id
              group asset by new {assetCategory.Id,assetCategory.CategoryName} into summary
              select new AssetCategorySummary
              {
                  CategoryId   = summary.Key.Id,
                  CategoryName = summary.Key.Name,
                  TotalAsset   = summary.Count()
              }).ToListAsync();
如果对实体进行修改,使eg资产具有类别属性,则查询可以简化为:

 var items = (from asset in Context.Assets
              group asset by new {asset.Category.Id,asset.Category.CategoryName} into summary
              select new AssetCategorySummary
              {
                  CategoryId   = summary.Key.Id,
                  CategoryName = summary.Key.Name,
                  TotalAsset   = summary.Count()
              }).ToListAsync();
这需要一些测试来确保它创建了一个合理的查询。过去有过一些惊喜,我没有时间检查最终EF Core 3.0中生成的SQL

更新

LINQPad 6可以使用EF Core 3,甚至可以使用外键约束从数据库生成DbContext

这个问题

 var items = (from asset in Context.Assets
              group asset by new {asset.Category.Id,asset.Category.CategoryName} into summary
              select new AssetCategorySummary
              {
                  CategoryId   = summary.Key.Id,
                  CategoryName = summary.Key.Name,
                  TotalAsset   = summary.Count()
              }).ToListAsync();

生成一个漂亮的SQL查询:

SELECT [a0].[ID] AS [CategoryId], [a0].[CategoryName], COUNT(*) AS [TotalAsset]
FROM [Assets] AS [a]
INNER JOIN [AssetCategories] AS [a0] ON [a].[CategoryID] = [a0].[ID]
GROUP BY [a0].[ID], [a0].[CategoryName]

使用join生成相同的SQL查询。

您仍然可以通过客户机评估在客户机上执行任何类型的set操作,只需在执行set操作之前插入一个AsEnumerable即可。在3.0之前的版本中,所有设置操作都是这样处理的,根据具体的用例,客户端评估可能与服务器评估一样有效。

自动客户端评估已经完全消失了-这不是一个选项,无法启用。是的,请参阅第二个链接中的缓解部分-如果查询无法完全翻译,请以可以翻译的形式重写查询,或者使用可计算、ToList或类似方法将数据显式带回客户端,然后使用LINQ to对象对其进行进一步处理。正如您所看到的,没有启用它的选项。此外,QueryClientEvaluationWarning被标记为[不再支持过时的自动客户端评估。不再生成此事件。]我没有得到此重大更改。即使是一个简单的GroupBy现在也失败了,这不是一个问题。升级到EF core 3时需要进行更改。除非有实际问题,否则不需要在这里报告。@s.Aziz Kazdal是的!这就是我告诉过你的,将数据提取到内存中,然后进行查询,虽然这样做并不有效,因为这会降低性能,或者在出现大量数据时会失败。@TanvirArjel需要一种额外的方法来实现这一点。有什么东西可以将原始sql写入数据库吗?不是实体。一些上下文数据库。RawSql@S.AzizKazdal您可以编写原始sql并使用FromSql方法。更多详情:@TanvirArjel no FromSql这不是我想要的。我想执行此查询并获取要列出的值。选择Kategori.Id,Kategori.CategoryName,countVarlik.Id作为[Varlık Sayısı]从Kategori.Id=Varlik.CategoryId组的Kategori.Id上的资产Varlik内部连接资产类别Kategori.CategoryId
SELECT [a0].[ID] AS [CategoryId], [a0].[CategoryName], COUNT(*) AS [TotalAsset]
FROM [Assets] AS [a]
INNER JOIN [AssetCategories] AS [a0] ON [a].[CategoryID] = [a0].[ID]
GROUP BY [a0].[ID], [a0].[CategoryName]