Joh*_* L. 3 c# sql linq sql-server entity-framework
我想获取客户 ID 列表以及该客户下的订单数量。过滤条件为:
所以,我会在 SQL 中执行以下操作:
SELECT customerID, COUNT(*)
FROM Orders
WHERE orderTotal > 10
GROUP BY customerID
HAVING COUNT(*) > 2
Run Code Online (Sandbox Code Playgroud)
在 EF 中,我认为这可以表示为:
dbContext.Order
.Where(o => o.orderTotal > 10)
.GroupBy(o => o.customerID)
.Where(g => g.Count() > 2)
.ToList();
Run Code Online (Sandbox Code Playgroud)
但这会产生以下 SQL,它使用派生表和连接而不是简单地使用HAVING子句。我认为这在性能方面远非最佳。是否有更好的方法在 EF 中制定案例,以便翻译后的查询将使用该HAVING条款?
SELECT
[Project1].[C1] AS [C1],
[Project1].[customerID] AS [customerID],
[Project1].[C2] AS [C2],
[Project1].[ID] AS [ID],
FROM ( SELECT
[GroupBy1].[K1] AS [customerID],
1 AS [C1],
[Extent2].[ID] AS [ID],
CASE WHEN ([Extent2].[storeID] IS NULL) THEN CAST(NULL AS int) ELSE 1 END AS [C2]
FROM (SELECT
[Extent1].[customerID] AS [K1],
COUNT(1) AS [A1]
FROM [dbo].[Orders] AS [Extent1]
WHERE [Extent1].[orderTotal] > cast(10 as decimal(18))
GROUP BY [Extent1].[customerID] ) AS [GroupBy1]
LEFT OUTER JOIN [dbo].[Orders] AS [Extent2] ON ([Extent2].[orderTotal] > cast(10 as decimal(18))) AND (([GroupBy1].[K1] = [Extent2].[customerID]) OR (([GroupBy1].[K1] IS NULL) AND ([Extent2].[customerID] IS NULL)))
WHERE [GroupBy1].[A1] > 2
) AS [Project1]
ORDER BY [Project1].[customerID] ASC, [Project1].[C2] ASC
Run Code Online (Sandbox Code Playgroud)
嗯,LINQ to Entities 查询并不等同于 SQL 查询,因为它返回一个分组列表(键和匹配元素对),而该列表根本没有 SQL 等效项。
如果在 SQL 查询中只返回 thecustomerId和 the Count:
db.Orders
.Where(o => o.orderTotal > 10)
.GroupBy(o => o.customerID)
.Select(g => new { customerId = g.Key, orderCount = g.Count() })
.Where(g => g.Count > 2)
.ToList();
Run Code Online (Sandbox Code Playgroud)
那么 EF 生成的 SQL 将与预期的几乎相同(或在功能上等同于):
db.Orders
.Where(o => o.orderTotal > 10)
.GroupBy(o => o.customerID)
.Select(g => new { customerId = g.Key, orderCount = g.Count() })
.Where(g => g.Count > 2)
.ToList();
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
1143 次 |
| 最近记录: |