-
Notifications
You must be signed in to change notification settings - Fork 13k
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
Prefer partition_point
to look up assoc items
#86392
Conversation
(rust-highfive has picked a reviewer for you, use r? to override) |
@bors try @rust-timer queue |
Awaiting bors try build completion. @rustbot label: +S-waiting-on-perf |
⌛ Trying commit c0efd2a with merge 05fd6aaa737fb22c7a1029b309aa3709bad95d72... |
☀️ Try build successful - checks-actions |
Queued 05fd6aaa737fb22c7a1029b309aa3709bad95d72 with parent 444a85a, future comparison URL. |
Finished benchmarking try commit (05fd6aaa737fb22c7a1029b309aa3709bad95d72): comparison url. Benchmarking this pull request likely means that it is perf-sensitive, so we're automatically marking it as not fit for rolling up. Please note that if the perf results are neutral, you should likely undo the rollup=never given below by specifying Importantly, though, if the results of this run are non-neutral do not roll this PR up -- it will mask other regressions or improvements in the roll up. @bors rollup=never |
Some small wins up to ~0.6% and the worst regression is regex-opt's 0.3%, I guess it's acceptable? |
@bors r+ |
📌 Commit c0efd2a has been approved by |
// FIXME: This should be in the standard library as `equal_range`. See rust-lang/rfcs#2184. | ||
match self.binary_search_idx(key) { | ||
Err(_) => self.idxs_to_items_enumerated(&[]), | ||
|
||
Ok(idx) => { | ||
let start = self.find_lower_bound(key, idx); | ||
let end = self.find_upper_bound(key, idx); | ||
let start = self.idx_sorted_by_item_key[..idx] | ||
.partition_point(|&i| self.items[i].0.borrow() != key); | ||
let end = idx | ||
+ self.idx_sorted_by_item_key[idx..] | ||
.partition_point(|&i| self.items[i].0.borrow() == key); | ||
self.idxs_to_items_enumerated(&self.idx_sorted_by_item_key[start..end]) |
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.
If I understand correctly, this is now doing a binary_search_by
to use a binary search to find any matching key, and then does two more binary searches (left and right of the item that was found), to find the first one. Why not just use a partition_point(|| .. < ..)
to find the first key directly?
That is, if you just remove the match and the Err case, and change the !=
to <
, it still works and does the same thing, right?
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 unless there's tons of duplicated keys, it can be a lot faster to only use partition_point for finding the lower bound, and just search linearly for the upper bound. See also the comment on line 128 before the change.
Edit: Oh, since this just returns an impl Iterator
, it can just not look for the upper_bound at all and return .take_while(..)
on the slice starting at the lower bound.)
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.
Oh neat, indeed we could improve it more. I'll open a PR to follow up it, it'd be great if you could review it, thanks!
☀️ Test successful - checks-actions |
…li-obk Improve `get_by_key_enumerated` more Follow-up of rust-lang#86392, this applies the suggestions by `@m-ou-se.` r? `@m-ou-se`
Since we now have
partition_point
(instead ofequal_range
), I think it's worth trying to use it instead of manually finding it.partition_point
usesbinary_search_by
internally (#85406) and its performance has been improved (#74024), so I guess this will make a performance difference.