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

Introduce LinkedList::drain_filter #46262

Merged
merged 2 commits into from
Nov 28, 2017
Merged

Introduce LinkedList::drain_filter #46262

merged 2 commits into from
Nov 28, 2017

Conversation

udoprog
Copy link
Contributor

@udoprog udoprog commented Nov 25, 2017

This introduces LinkedList::remove_if.

This operation embodies one of the use-cases where LinkedList would typically be preferred over Vec: random removal and retrieval.

There are a number of considerations with this:

Should there be two remove_if methods? One where elements are only removed, one which returns a collection of removed elements.

Should this be implemented using a draining iterator pattern that covers both cases? I suspect that would incur a bit of overhead (moving the element into the iterator, then into a new collection). But I'm not sure. Maybe that's an acceptable compromise to maximize flexibility.

I don't feel I've had enough exposure to unsafe programming in rust to be certain the implementation is correct. This relies quite heavily on moving around copies of Shared pointers to make the code reasonable. Please help me out :).

@rust-highfive
Copy link
Contributor

Thanks for the pull request, and welcome! The Rust team is excited to review your changes, and you should hear from @kennytm (or someone else) soon.

If any changes to this PR are deemed necessary, please add them as extra commits. This ensures that the reviewer can see what has changed since they last reviewed the code. Due to the way GitHub handles out-of-date commits, this should also make it reasonably obvious what issues have or haven't been addressed. Large or tricky changes may require several passes of review and changes.

Please see the contribution instructions for more information.

@kennytm kennytm added S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. T-libs-api Relevant to the library API team, which will review and decide on the PR/issue. labels Nov 25, 2017
@kennytm
Copy link
Member

kennytm commented Nov 25, 2017

This PR introduces a new method. Deferring the decision to the libs team.

r? @dtolnay

@rust-highfive rust-highfive assigned dtolnay and unassigned kennytm Nov 25, 2017
Copy link
Member

@dtolnay dtolnay left a comment

Choose a reason for hiding this comment

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

Vec::drain_filter is a much more general and equally efficient form of remove_if where the caller can do whatever they want with the removed values (building a new LinkedList of them, as with remove_if, seems like an uncommon use case). The equivalent of your example code would be:

let mut d = LinkedList::new();
d.push_back(1);
d.push_back(2);
d.push_back(3);

let mut removed = LinkedList::new();
for node in d.drain_filter(|v| *v < 3) {
    removed.push_back(node);
}
assert_eq!(removed.len(), 2);
assert_eq!(d.len(), 1);

I would prefer to be consistent by providing the same API for LinkedList.

@udoprog
Copy link
Contributor Author

udoprog commented Nov 25, 2017

@dtolnay will do, that also answers most of my questions. Thanks!

@udoprog
Copy link
Contributor Author

udoprog commented Nov 25, 2017

Relates to rust-lang/rfcs#2140

Relates to rust-lang/rfcs#2140 - drain_filter for all collections

`drain_filter` is implemented instead of `LinkedList::remove_if` based
on review feedback.
@udoprog
Copy link
Contributor Author

udoprog commented Nov 25, 2017

I ported all the tests on Vec for drain_filter. Would be nice to abstract them, but I felt that would be a change that is a bit too large for my taste.

All changes are now also associated with the drain_filter feature.

@udoprog udoprog changed the title Introduce LinkedList::remove_if Introduce LinkedList::drain_filter Nov 25, 2017
@bluss
Copy link
Member

bluss commented Nov 25, 2017

Consistency is important. Note however that Linked List can do this kind of operation -- remove some elements during the scan and string them together a resulting linked list of the removed elements -- without any de/allocation of list nodes or even any moving of element values. It would be good to expose that functionality as well, but it is kind of its own kind of operation.

@udoprog
Copy link
Contributor Author

udoprog commented Nov 25, 2017

The original patch did something like that. It could at some point be resurrected as a method on DrainFilter if it's deemed important.

I'd be curious to see how well the optimizer deals with the example provided by @dtolnay. I wouldn't be surprised if specialized methods aren't needed.

@bluss
Copy link
Member

bluss commented Nov 26, 2017

This could be covered with specialization. Imagine LinkedList::extend(into iter) and LinkedList::extend(drain_filter) (extend brings collect with it implicitly) both can get this specialization, to just reuse the list nodes. It's got an appeal of not needing any additional API design and it hits consistency as well.

@udoprog
Copy link
Contributor Author

udoprog commented Nov 27, 2017

Since the extend approach is compatible with a draining iterator it can be introduced separately.

@dtolnay With this in mind, can I get another round of review?

@dtolnay
Copy link
Member

dtolnay commented Nov 27, 2017

@bors r+

@bors
Copy link
Collaborator

bors commented Nov 27, 2017

📌 Commit 60aa834 has been approved by dtolnay

@kennytm kennytm added S-waiting-on-bors Status: Waiting on bors to run and complete tests. Bors will change the label on completion. and removed S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. labels Nov 27, 2017
kennytm added a commit to kennytm/rust that referenced this pull request Nov 27, 2017
…olnay

Introduce LinkedList::drain_filter

This introduces `LinkedList::remove_if`.

This operation embodies one of the use-cases where `LinkedList` would typically be preferred over `Vec`: random removal and retrieval.

There are a number of considerations with this:

Should there be two `remove_if` methods? One where elements are only removed, one which returns a collection of removed elements.

Should this be implemented using a draining iterator pattern that covers both cases? I suspect that would incur a bit of overhead (moving the element into the iterator, then into a new collection). But I'm not sure. Maybe that's an acceptable compromise to maximize flexibility.

I don't feel I've had enough exposure to unsafe programming in rust to be certain the implementation is correct. This relies quite heavily on moving around copies of Shared pointers to make the code reasonable. Please help me out :).
bors added a commit that referenced this pull request Nov 27, 2017
Rollup of 10 pull requests

- Successful merges: #45506, #46174, #46231, #46240, #46249, #46258, #46262, #46275, #46282, #46285
- Failed merges:
@bors bors merged commit 60aa834 into rust-lang:master Nov 28, 2017
@udoprog udoprog deleted the linked-list-remove-if branch November 28, 2017 11:31
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
S-waiting-on-bors Status: Waiting on bors to run and complete tests. Bors will change the label on completion. T-libs-api Relevant to the library API team, which will review and decide on the PR/issue.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants