实体框架6加入了太多

时间:2013-12-22 19:13:43

标签: c# sql linq entity-framework

我有以下LINQ查询:

var query = from a in c.ArticleSet
            where a.GlobalAccess == false && 
                  a.Published == true && 
                  a.MagazineSet.IsPublished == true && 
                  a.MagazineSet.Press_Id == pressId && 
                  a.Tests.Id==a.Id && 
                  a.Tests.IsDeleted == false && 
                  a.Tests.IsPublished == true
            orderby a.Id descending
            select a.Id;

它像这样转换为SQL:

ADO.NET:Execute Reader "SELECT 
[Project1].[Id] AS [Id]
FROM ( SELECT 
    [Filter3].[Id1] AS [Id]
    FROM   (SELECT [Filter2].[Id1], [Filter2].[Magazine_Id1], [Filter2].[Press_Id1], [Filter2].[Press_Id2], [Filter2].[Press_Id3]
        FROM    (SELECT [Filter1].[Id1], [Filter1].[Magazine_Id1], [Extent5].[Press_Id] AS [Press_Id1], [Extent6].[Press_Id] AS [Press_Id2], [Extent7].[Press_Id] AS [Press_Id3]
            FROM       (SELECT [Extent1].[Id] AS [Id1], [Extent1].[Magazine_Id] AS [Magazine_Id1]
                FROM   [dbo].[ArticleSet] AS [Extent1]
                INNER JOIN [dbo].[Tests] AS [Extent2] ON ([Extent1].[Id] = [Extent2].[Id]) AND ([Extent2].[Id] = [Extent1].[Id])
                INNER JOIN [dbo].[MagazineSet] AS [Extent3] ON [Extent1].[Magazine_Id] = [Extent3].[Id]
                WHERE (0 = [Extent1].[GlobalAccess]) AND (1 = [Extent1].[Published]) AND (1 = [Extent3].[IsPublished]) ) AS [Filter1]
            INNER JOIN [dbo].[MagazineSet] AS [Extent4] ON [Filter1].[Magazine_Id1] = [Extent4].[Id]
            LEFT OUTER JOIN [dbo].[MagazineSet] AS [Extent5] ON [Filter1].[Magazine_Id1] = [Extent5].[Id]
            LEFT OUTER JOIN [dbo].[MagazineSet] AS [Extent6] ON [Filter1].[Magazine_Id1] = [Extent6].[Id]
            LEFT OUTER JOIN [dbo].[MagazineSet] AS [Extent7] ON [Filter1].[Magazine_Id1] = [Extent7].[Id]
            INNER JOIN [dbo].[Tests] AS [Extent8] ON [Filter1].[Id1] = [Extent8].[Id]
            WHERE 0 = [Extent8].[IsDeleted] ) AS [Filter2]
        INNER JOIN [dbo].[Tests] AS [Extent9] ON [Filter2].[Id1] = [Extent9].[Id]
        INNER JOIN [dbo].[Tests] AS [Extent10] ON [Filter2].[Id1] = [Extent10].[Id]
        WHERE 1 = [Extent10].[IsPublished] ) AS [Filter3]
    INNER JOIN [dbo].[Tests] AS [Extent11] ON [Filter3].[Id1] = [Extent11].[Id]
    WHERE (([Filter3].[Press_Id1] = @p__linq__0) AND ( NOT ([Filter3].[Press_Id2] IS NULL OR @p__linq__0 IS NULL))) OR (([Filter3].[Press_Id3] IS NULL) AND (@p__linq__0 IS NULL))
)  AS [Project1]
ORDER BY [Project1].[Id] DESC"

命令文本

"SELECT [Project1].[Id] AS [Id]
FROM ( SELECT 
    [Filter3].[Id1] AS [Id]
    FROM   (SELECT [Filter2].[Id1], [Filter2].[Magazine_Id1], [Filter2].[Press_Id1], [Filter2].[Press_Id2], [Filter2].[Press_Id3]
        FROM    (SELECT [Filter1].[Id1], [Filter1].[Magazine_Id1], [Extent5].[Press_Id] AS [Press_Id1], [Extent6].[Press_Id] AS [Press_Id2], [Extent7].[Press_Id] AS [Press_Id3]
            FROM       (SELECT [Extent1].[Id] AS [Id1], [Extent1].[Magazine_Id] AS [Magazine_Id1]
                FROM   [dbo].[ArticleSet] AS [Extent1]
                INNER JOIN [dbo].[Tests] AS [Extent2] ON ([Extent1].[Id] = [Extent2].[Id]) AND ([Extent2].[Id] = [Extent1].[Id])
                INNER JOIN [dbo].[MagazineSet] AS [Extent3] ON [Extent1].[Magazine_Id] = [Extent3].[Id]
                WHERE (0 = [Extent1].[GlobalAccess]) AND (1 = [Extent1].[Published]) AND (1 = [Extent3].[IsPublished]) ) AS [Filter1]
            INNER JOIN [dbo].[MagazineSet] AS [Extent4] ON [Filter1].[Magazine_Id1] = [Extent4].[Id]
            LEFT OUTER JOIN [dbo].[MagazineSet] AS [Extent5] ON [Filter1].[Magazine_Id1] = [Extent5].[Id]
            LEFT OUTER JOIN [dbo].[MagazineSet] AS [Extent6] ON [Filter1].[Magazine_Id1] = [Extent6].[Id]
            LEFT OUTER JOIN [dbo].[MagazineSet] AS [Extent7] ON [Filter1].[Magazine_Id1] = [Extent7].[Id]
            INNER JOIN [dbo].[Tests] AS [Extent8] ON [Filter1].[Id1] = [Extent8].[Id]
            WHERE 0 = [Extent8].[IsDeleted] ) AS [Filter2]
        INNER JOIN [dbo].[Tests] AS [Extent9] ON [Filter2].[Id1] = [Extent9].[Id]
        INNER JOIN [dbo].[Tests] AS [Extent10] ON [Filter2].[Id1] = [Extent10].[Id]
        WHERE 1 = [Extent10].[IsPublished] ) AS [Filter3]
    INNER JOIN [dbo].[Tests] AS [Extent11] ON [Filter3].[Id1] = [Extent11].[Id]
    WHERE (([Filter3].[Press_Id1] = @p__linq__0) AND ( NOT ([Filter3].[Press_Id2] IS NULL OR @p__linq__0 IS NULL))) OR (([Filter3].[Press_Id3] IS NULL) AND (@p__linq__0 IS NULL))
)  AS [Project1]
ORDER BY [Project1].[Id] DESC

我认为在这个例子中3连接就足够了。 为什么Entity框架会创建如此庞大的查询?

1 个答案:

答案 0 :(得分:0)

你的linq查询很好,除了它一次引用了很多表。如果这真的是你想要的,那就没有什么可怕的了。它只是“超级高效”,因为你一次引用3个表。

请记住,EF生成的SQL查询是机器创建的。它并不总是最好的,但它应该完全有效。我不确定为什么它选择这种形式而不是另一种形式,但如果它有效,为什么要担心?

我总是建议人们将查询限制为只需要他们所需的引用表。随着系统的增长,处理时间也会增加,并且通过这么多表连接,它可能会开始变得更加明显。