Entity framework EF Core 2.2 LINQ查询在EF Core 3.0中不工作
以下代码在EF Core 2.2上运行良好,但不在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
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]