-
Notifications
You must be signed in to change notification settings - Fork 3.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Optimize multiple consecutive LIMIT
s
#35384
Changes from all commits
02f8d6f
3bbb726
9cdc7a2
08c8569
3e3c695
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1924,6 +1924,16 @@ public void ApplyLimit(SqlExpression sqlExpression) | |
PushdownIntoSubquery(); | ||
} | ||
|
||
SetLimit(sqlExpression); | ||
} | ||
|
||
/// <summary> | ||
/// Sets a new limit of the <see cref="SelectExpression" /> to limit the number of rows returned in the result set. | ||
/// </summary> | ||
/// <param name="sqlExpression">An expression representing limit row count.</param> | ||
[EntityFrameworkInternal] | ||
public void SetLimit(SqlExpression sqlExpression) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Please add [EntityFrameworkInternal] as while this looks good for now, I don't see this API sticking around (e.g. once SelectExpression is immutable). There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Indeed I assume that making There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Absolutely... That's high up on my list for 10. |
||
{ | ||
Limit = sqlExpression; | ||
|
||
if (Offset is null && Limit is SqlConstantExpression { Value: 1 }) | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -299,17 +299,15 @@ public override async Task Skip_navigation_order_by_single_or_default(bool async | |
""" | ||
SELECT [s0].[Id], [s0].[Name] | ||
FROM [EntityOnes] AS [e] | ||
OUTER APPLY ( | ||
SELECT TOP(1) [s].[Id], [s].[Name] | ||
LEFT JOIN ( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Interesting change... The LEFT JOIN is obviously better than the OUTER APPLY (and also unlocks SQLite which doesn't support APPLY), but then we get the row number inside... Looks OK in any case. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I am afraid we have that in multiple places; I might eventually investigate, but seems like something that is already happening :\ There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah. BTW CROSS/OUTER APPLY vs. ROW_NUMBER() is something that's already tracked in #30450, it's the kind of deep performance investigations we haven't gotten around to yet. I hope that once the basic query architecture refactoring is farther ahead, all this will be much easier to tackle. |
||
SELECT [s].[Id], [s].[Name], [s].[LeftId] | ||
FROM ( | ||
SELECT TOP(1) [e0].[Id], [e0].[Name] | ||
SELECT [e0].[Id], [e0].[Name], [j].[LeftId], ROW_NUMBER() OVER(PARTITION BY [j].[LeftId] ORDER BY [e0].[Id]) AS [row] | ||
FROM [JoinOneSelfPayload] AS [j] | ||
INNER JOIN [EntityOnes] AS [e0] ON [j].[RightId] = [e0].[Id] | ||
WHERE [e].[Id] = [j].[LeftId] | ||
ORDER BY [e0].[Id] | ||
) AS [s] | ||
ORDER BY [s].[Id] | ||
) AS [s0] | ||
WHERE [s].[row] <= 1 | ||
) AS [s0] ON [e].[Id] = [s0].[LeftId] | ||
"""); | ||
} | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
we could actually aim for never doing a pushdown and instead replace the limit with
LEAST(Limit, sqlExpression
, but I am unsure if this is easy to do here (in this context we don't have access toTranslateMin
)There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
And you can't guarantee
LEAST
is supported at this stageThere was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
exactly :(
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am experimenting with:
SelectExpression.SetLimit
methodApplyLimit
invocations withSetLimit
s as they are done from a context that also invokesGenerateLeast
(and if that returnsnull
, I can easily fall back toApplyLimit
)There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm probably in the minority but it probably will break EFCore.Jet. I can only have a constant expression for TOP. And since I don't have LIMIT/OFFSET, TOP= LIMIT. And I do have a translation for LEAST
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
uhm... how does that work with non-constant
Take()
? (parameters are likely the most common, but in theory any kind of expression is allowed there)There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also, note that as long as the code is only
LIMIT
ing on constants, you still get a (combined) constant as the newLIMIT
.LEAST
/MIN
are emitted when combining non-constants (or a constant and a non-constant), so it might be possible to handle it in the same way as you are doing now (I would assume something with row numbers)There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I throw a Not Translated error (no row_number or equivlalent in ms access).
But I'm happy as long as a constant expression stays as a plain constant expression
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just before I sent it off to Oledb/ODBC, the query is modified to inline the parameter. If it refers to a column, well it just fails
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Guys, one of my major gripes with the current query pipeline design is the non-openness of SelectExpression, i.e. code there isn't overridable by providers; this case with integrating LEAST() instead of pushing down is a great example of why that's needed. I do have a plan in mind: I'd like to first make SelectExpression fully mutable, and then extract most/all actual logic out of it, so that SelectExpression becomes a regular, pure, immutable expression just like any other. The logic could go to QueryableMethodTranslatingEV, but some of these operations are probably also needed in postprocessing, so possible a separate service is maybe needed, that would be accessible from everywhere and be overridable by the provider. I hope to make at least some progress in this general direction in 10 (e.g. making SelectExpression fully immutable).
Until then, we're a bit stuck hacking around... @ranma42 what you've done in this draft looks good to me - let me know when this is ready for review.