Skip to content
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

Add a simple queue implementation with better performance than Array.shift #49623

Merged
merged 2 commits into from
Jun 24, 2022

Conversation

amcasey
Copy link
Member

@amcasey amcasey commented Jun 21, 2022

This lets us clean up the hack introduced in #49581

@typescript-bot typescript-bot added Author: Team For Uncommitted Bug PR for untriaged, rejected, closed or missing bug labels Jun 21, 2022
@nievesmiguel
Copy link

amcasey:Queue

Co-authored-by: Mateusz Burzyński <[email protected]>
@captain-yossarian
Copy link

@amcasey Hi, could you please explain your changes on a small example with context of this PR I'm asking, because I don't know how TS works under the hood but your change looks very interesting. Thank you

@amcasey
Copy link
Member Author

amcasey commented Jun 24, 2022

@captain-yossarian Neither change is particularly specific to TS. Basically, when you shift and element off the beginning of an array, you have to re-number all the remaining elements. If the array is very large, this can take a long time. If the array has n elements and you shift each of them one-by-one, you'll have to re-number (n-1) the first time, (n-2) the second time, (n-3) the third time, etc. This is referred to as a "quadratic" (i.e. slow) algorithm because the sum is roughly n^2. Both this change and the one it replaces were eliminating individual shifts in favor of larger deletions. So, in round numbers, the first deletion removes n/2 elements, the second deletion removes n/4 elements, etc. This is "linear", rather than quadratic, because the sum is roughly n. Hope that helps!

@captain-yossarian
Copy link

@amcasey Thank you very much for explanation

@amcasey amcasey merged commit 020ef41 into microsoft:main Jun 24, 2022
@amcasey amcasey deleted the Queue branch June 24, 2022 17:07
const q = [node];
while (q.length && i < 100) {
const q = createQueue<Expression>();
q.enqueue(node);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To me this part of the code seems like that order wouldnt matter so always using last element so i wonder if always using last element and keeping tail thats used to add more items and sets array.length to compact array once in a while might be better.
Thoughts?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I didn't read it carefully when I made the change - I just checked that it would behave the same after. Reading it now, it looks like switching from a queue to a stack would change a BFS to a DFS, which may or may not be desirable.

@@ -504,18 +504,18 @@ namespace ts.server {
// If `getResultsForPosition` returns results for a project, they go in here
const resultsMap = new Map<Project, readonly TResult[]>();

const queue: ProjectAndLocation[] = [];
const queue = createQueue<ProjectAndLocation>();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Even here, does order matter if not we could get away with having to move array elements right?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe order does matter, though that doesn't necessarily mean we couldn't accomplish the same thing with a stack.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why does order matter ? in all find all references etc when aggregating results between project, the order shouldnt matter right ? Yes our baselines may change but from end to end result perspective order shouldnt matter ?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe the search order affects the value of isDefinition. I believe it's also the case that FAR is asymmetric (B being a reference to A, does not imply that A is a reference to B), but I think we've already decided to ignore that restriction, so it's probably just isDefinition. Now that isDefinition is computed in a post-pass, it's possible that we can drop the strict ordering, but I'm frankly pretty burned out on rewriting this code. I have no objections if you'd like to, but I'd suggest we wait until 4.9, now that we're in the RC period.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Author: Team For Uncommitted Bug PR for untriaged, rejected, closed or missing bug
Projects
None yet
Development

Successfully merging this pull request may close these issues.

8 participants