我正在尝试针对LINQ实现左外联机,以针对实体框架Core 2.0 DBContext。必须将查询转换为SQL,而不是在本地进行评估,这一点很重要。我已经回顾了几种堆叠的解决方案,其中包括该解决方案,这是很好的,但没有一个使用EF Core。
我遇到的问题是EF核心返回DefaultIfEmpty()
方法的以下警告/错误:
The LINQ expression 'DefaultIfEmpty()' could not be translated and will be evaluated locally
没有DefaultIfEmpty()
方法,使用内部连接。我的linq查询看起来像这样:
var join = context.Portfolios
.Where(p => p.IsActive)
.GroupJoin(context.BankAccounts,
prt => prt.Id,
bnk => bnk.PortfolioId,
(prt, bnks) => new {Portfolio=prt,Account=bnks.DefaultIfEmpty()})
.SelectMany(r => r.Accounts.DefaultIfEmpty(),
(p, b) => new
{
Id = p.Portfolio.Id,
BankAccount = b.BankAccountNumber,
BankRef = b.BeneficiaryReference,
Code = p.Portfolio.Code,
Description = p.Portfolio.DisplayName
});
有人知道解决这个问题吗?
好吧,这是我的错误,基于另一个问题的评论,该评论指出DefaultIfEmpty()
是使查询成为外部连接的必要条件。查看基础SQL,当我删除DefaultIfEmpty()
规范时,将向数据库提交左JOIN 。我不确定这是否与在内存中的左JON JON COMN COMPONTIONS有所不同,但它解决了我的问题。
由EF Core生成的SQL为此Linq查询看起来像这样:
SELECT [p].[ID],
[bnk].[BankAccountNumber] AS [BankAccount],
[bnk].[BeneficiaryReference] AS [BankRef],
[p].[Code],
[p].[DisplayName] AS [Description]
FROM [Portfolio] AS [p]
LEFT JOIN [BankAccount] AS [bnk] ON [p].[ID] = [bnk].[PortfolioId]
WHERE (([p].[IsActive] = 1)))
编辑:找到该测试的时间,@ivan Stoev是正确的:如果您的导航属性在EF上下文定义中正确设置,则EF将生成左JON。使用EF时,这是一种更好的方法。
ef导航属性:
public virtual ICollection<BankAccount> BankAccounts { get; set; }
通过导航属性查询:
var join = context.Portfolios
.Where(p => p.IsActive)
.SelectMany(p => p.BankAccounts.DefaultIfEmpty(), (p, b) => new
{
Id = p.Id,
BankAccount = b.BankAccountNumber,
BankRef = b.BeneficiaryReference,
Code = p.Code,
Description = p.DisplayName
});
结果SQL代码:
SELECT [p].[ID], [p.BankAccounts].[BankAccountNumber] AS [BankAccount], [p.BankAccounts].[BeneficiaryReference] AS [BankRef], [p].[Code], [p].[DisplayName] AS [Description]
FROM [core].[Portfolio] AS [p]
LEFT JOIN [ims].[BankAccount] AS [p.BankAccounts] ON [p].[ID] = [p.BankAccounts].[PortfolioId]
WHERE (([p].[IsActive] = 1))
请注意,从LINQ查询中删除DefaultIfEmpty()
会导致内部联接。