一种找出和测试自己的方法:
打开 SQL Server Management Studio,打开一个新查询,选择 EF 将运行的数据库并运行此查询:
SELECT top 10 deqs.last_execution_time AS [Time], dest.TEXT AS [Query]
FROM sys.dm_exec_query_stats AS deqs
CROSS APPLY sys.dm_exec_sql_text(deqs.sql_handle) AS dest
ORDER BY deqs.last_execution_time DESC
这会告诉您过去对数据库运行的 10 个查询。
在 TestWhenSqlFires() 的第一行设置断点,运行您的代码,然后在遍历每一行后运行上述查询。你会找到:
// C# Line 1
var db = new Db();
--SQL Line 1
SELECT TABLE_SCHEMA SchemaName, TABLE_NAME Name FROM INFORMATION_SCHEMA.TABLES
WHERE TABLE_TYPE = 'BASE TABLE'
// C# Line 2
var items = db.SimpleObjects;
--SQL Line 2
SELECT COUNT(*) FROM [sys].[databases] WHERE [name]=@1
SELECT [GroupBy1].[A1] AS [C1] FROM (
SELECT COUNT(1) AS [A1] FROM [dbo].[__MigrationHistory] AS [Extent1]
) AS [GroupBy1]
(@1 nvarchar(4000))SELECT TOP (1) [Project1].[C1] AS [C1],
[Project1].[MigrationId] AS [MigrationId],
[Project1].[Model] AS [Model] FROM (
SELECT [Extent1].[MigrationId] AS [MigrationId],
[Extent1].[Model] AS [Model], 1 AS [C1]
FROM [dbo].[__MigrationHistory] AS [Extent1]
) AS [Project1] ORDER BY [Project1].[MigrationId] DESC
// C# Line 3
var byId = WhereId(items, 1);
--SQL Line 3
// C# Line 4
var array = byId.ToArray();
--SQL Line 4
SELECT [Extent1].[Id] AS [Id], [Extent1].[Stuff] AS [Stuff]
FROM [dbo].[SimpleObject] AS [Extent1]
最终的 SQL 查询是 EF 实际获取数据。之前的查询只是在预热 - 验证数据库是否存在,它正在使用 EF5 迁移历史,并且它与当前的迁移历史哈希匹配。
所以答案是 - 第 4 行,之后.ToArray()
被调用(或任何枚举集合的调用,如 .ToList()、foreach 等)。值得注意的是,将其传递给接受 IEnumerable 的方法,即使涉及特定的 Generic,也不会枚举集合,因此不会在必要之前触发 SQL。