繁体   English   中英

使用连接和列别名的分页顺序时,EF Core 2中的Linq查询不起作用

[英]Linq query in EF Core 2 when using joins and pagination order by column alias doesn't work

目前,我们正在尝试使用LinqEF Core 2将排序写出到服务器端分页中。 我们遇到了一个问题,即使用分页时Linq生成的列别名不起作用。 但是,如果我们不分页,它将按预期工作。

输出的查询中的所有列都是别名,因为我们在模型中具有不同的属性名称,而数据库的列名称也不同,但这对我们的知识没有影响。

这是没有分页的Linq查询:

var source = from p in _ppmRepository.GetAll()
                         join jt in _jobTypeRepository.GetAll() on p.PpmFkeyInSeq equals jt.Id into jtdata
                         from jt in jtdata.DefaultIfEmpty()
                         join a in _assetRepository.GetAll() on p.PpmFkeyArSeq equals a.Id into aData
                         from a in aData.DefaultIfEmpty()
                         where p.PpmFkeyBgSeq == bldId
                         orderby p.PpmFreq
                         select new BuildingPpmListViewModel
                         {
                             PpmId = p.Id,
                             PpmFreq = p.PpmFreq,
                             PpmNextService = p.PpmNextService,
                             TotalCost = p.TotalCost,
                             PpmPeriodUnits = p.PpmPeriodUnits,
                             PpmFkeyPriDesc = p.PpmFkeyPriDesc,
                             JtTitle = jt.JtTitle,
                             AssetId = p.PpmFkeyArSeq,
                             AssetDescription = a.AssetDescription,
                             IsDeleted = p.IsDeleted

                         };
            source = source.Where(i => i.JtTitle.Contains("audit") && i.AssetDescription.Contains("df"));

这是由ef core产生的输出查询,该查询有效:

SELECT [p].[PPM_SEQ] AS [PpmId], [p].[PPM_FREQ] AS [PpmFreq], [p].[PPM_NEXT_SERVICE] AS [PpmNextService], 
CAST([p].[TotalCost] AS float) AS [TotalCost], [p].[PPM_PERIOD_UNITS] AS [PpmPeriodUnits], [p].[PPM_FKEY_PRI_DESC] AS [PpmFkeyPriDesc], 
[t].[jt_title] AS [JtTitle], [p].[PPM_FKEY_AR_SEQ] AS [AssetId], [t0].[AR_DESCRIPTION] AS [AssetDescription], [p].[Deleted] AS [IsDeleted]
FROM [PPMs] AS [p]
LEFT JOIN (
    SELECT [j].*
    FROM [JobTypes] AS [j]
) AS [t] ON [p].[PPM_FKEY_IN_SEQ] = [t].[jt_seq]
LEFT JOIN (
    SELECT [a].*
    FROM [Assets] AS [a]
) AS [t0] ON [p].[PPM_FKEY_AR_SEQ] = [t0].[ar_seq]
WHERE ([p].[PPM_FKEY_BG_SEQ] = 172) AND ((CHARINDEX(N'audit', [t].[jt_title]) > 0) AND (CHARINDEX(N'df', [t0].[AR_DESCRIPTION]) > 0))
ORDER BY [PpmFreq]

这是带有分页的Linq查询:

var source = from p in _ppmRepository.GetAll()

                         join jt in _jobTypeRepository.GetAll() on p.PpmFkeyInSeq equals jt.Id into jtdata
                         from jt in jtdata.DefaultIfEmpty()
                         join a in _assetRepository.GetAll() on p.PpmFkeyArSeq equals a.Id into aData
                         from a in aData.DefaultIfEmpty()
                         where p.PpmFkeyBgSeq == bldId
                         orderby p.PpmFreq
                         select new BuildingPpmListViewModel
                         {
                             PpmId = p.Id,
                             PpmFreq = p.PpmFreq,
                             PpmNextService = p.PpmNextService,
                             TotalCost = p.TotalCost,
                             PpmPeriodUnits = p.PpmPeriodUnits,
                             PpmFkeyPriDesc = p.PpmFkeyPriDesc,
                             JtTitle = jt.JtTitle,
                             AssetId = p.PpmFkeyArSeq,
                             AssetDescription = a.AssetDescription,
                             IsDeleted = p.IsDeleted

                         };
            source = source.Where(i => i.JtTitle.Contains("audit") && i.AssetDescription.Contains("df")).Skip(0).Take(50);

这是分页的输出,其中PpmFreq在over函数顺序中是[p].[PPM_FREQ]的别名[p].[PPM_FREQ] SQL找不到:

SELECT [t1].[PpmId], [t1].[PpmFreq], [t1].[PpmNextService], [t1].[TotalCost], [t1].[PpmPeriodUnits], 
[t1].[PpmFkeyPriDesc], [t1].[JtTitle], [t1].[AssetId], [t1].[AssetDescription], [t1].[IsDeleted]
FROM (
    SELECT [p].[PPM_SEQ] AS [PpmId], [p].[PPM_FREQ] AS [PpmFreq], [p].[PPM_NEXT_SERVICE] AS [PpmNextService], 
    CAST([p].[TotalCost] AS float) AS [TotalCost], [p].[PPM_PERIOD_UNITS] AS [PpmPeriodUnits], [p].[PPM_FKEY_PRI_DESC] AS 
    [PpmFkeyPriDesc], [t].[jt_title] AS [JtTitle], [p].[PPM_FKEY_AR_SEQ] AS [AssetId], [t0].[AR_DESCRIPTION] AS [AssetDescription], 
    [p].[Deleted] AS [IsDeleted], ROW_NUMBER() OVER(ORDER BY [PpmFreq]) AS [__RowNumber__]
    FROM [PPMs] AS [p]
    LEFT JOIN (
        SELECT [j].*
        FROM [JobTypes] AS [j]
    ) AS [t] ON [p].[PPM_FKEY_IN_SEQ] = [t].[jt_seq]
    LEFT JOIN (
        SELECT [a].*
        FROM [Assets] AS [a]
    ) AS [t0] ON [p].[PPM_FKEY_AR_SEQ] = [t0].[ar_seq]
    WHERE (([p].[PPM_FKEY_BG_SEQ] = 172)) AND ((CHARINDEX(N'audit', [t].[jt_title]) > 0) 
    AND (CHARINDEX(N'df', [t0].[AR_DESCRIPTION]) > 0))
) AS [t1]
WHERE ([t1].[__RowNumber__] > 0) AND ([t1].[__RowNumber__] <= (50))

这似乎是问题的出处,因为我们可以对其稍加修改以从数据库中获得正确的结果:

ROW_NUMBER() OVER(ORDER BY [PpmFreq]) AS [__RowNumber__]

如果我们要修改上面的语句以将表别名也包含为[p].[PPM_FREQ] ,例如: ROW_NUMBER() OVER(ORDER BY [p].[PPM_FREQ]) AS [__RowNumber__]那么我们的问题就可以解决了,但对于我们当前的linq查询来说似乎不可能。

看看下面的效果是否更好:

            var source = (from p in _ppmRepository.GetAll()
                         join jt in _jobTypeRepository.GetAll() on p.PpmFkeyInSeq equals jt.Id into jtdata
                         from jt in jtdata.DefaultIfEmpty()
                         join a in _assetRepository.GetAll() on p.PpmFkeyArSeq equals a.Id into aData
                         from a in aData.DefaultIfEmpty()
                         select new BuildingPpmListViewModel
                         {
                             PpmId = p.Id,
                             PpBgSeq = p.PpmFkeyBgSeq,
                             PpmFreq = p.PpmFreq,
                             PpmNextService = p.PpmNextService,
                             TotalCost = p.TotalCost,
                             PpmPeriodUnits = p.PpmPeriodUnits,
                             PpmFkeyPriDesc = p.PpmFkeyPriDesc,
                             JtTitle = jt.JtTitle,
                             AssetId = p.PpmFkeyArSeq,
                             AssetDescription = a.AssetDescription,
                             IsDeleted = p.IsDeleted

                         })
                         .Where(x =>  x.PpBgSeq == bldId)
                         .OrderBy(x => x.PpmFreq)
                         .ToList();

这是一个已知问题,我们稍后将其直接提交给ef核心团队。

这是一个已知问题,已在即将发布的2.1版本中解决。您可以在此处查看更多详细信息和可能的解决方法github.com / aspnet / EntityFrameworkCore / issues / 9535` Smit Patel

如果您运行每晚构建,则可以解决上述问题。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM