-
Notifications
You must be signed in to change notification settings - Fork 13.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
Functions that exhaust iterators should take them by value. #13038
Comments
# Summary Changed `iter::Extendable` and `iter::FromIterator` to take a `Iterator` by value. These functions always exhaust the passed `Iterator`, and are often used for transferring the values of a new `Iterator` directly into a data structure, so using them usually require the use of the `&mut` operator: ``` foo.extend(&mut bar.move_iter()); // Transfer content from bar into foo let mut iter = ...; foo.extend(&mut iter); // iter is now empty ``` This patch changes both the `FromIterator` and `Extendable` traits to take the iterator by value instead, which makes the common case of using these traits less heavy: ``` foo.extend(bar.move_iter()); // Transfer content from bar into foo let iter = ...; foo.extend(iter); // iter is now inaccessible if it moved // or unchanged if it was Pod and copied. ``` # Composability This technically makes the traits less flexible from a type system pov, because they now require ownership. However, because `Iterator` provides the `ByRef` adapter, there is no loss of functionality: ``` foo.extend(iter.by_ref()); // Same semantic as today, for the few situations where you need it. ``` # Motivation This change makes it less painful to use iterators for shuffling values around between collections, which makes it more acceptable to always use them for this, enabling more flexibility. For example, `foo.extend(bar.move_iter())` can generally be the fastest way to append an collections content to another one, without both needing to have the same type. Making this easy to use would allow the removal of special cased methods like `push_all()` on vectors. (See #12456) I opened #13038 as well, to discuss this change in general if people object to it. # Further work This didn't change the `collect()` method to take by value `self`, nor any of the other adapters that also exhaust their iterator argument. For consistency this should probably happen in the long term, but for now this is too much trouble, as every use of them would need to be checked for accidentally changed semantic by going `&mut self -> self`. (which allows for the possibility that a `Pod` iterator got copied instead of exhausted without generating a type error by the change)
Are there any other methods that should be changed to take the iterator by value? I've seen that you already modified the |
Are there any methods that take Iterators by reference at all? You can always use |
Last time I checked, most iterator methods that where not adapters took by reference. |
Following the logic of the original issue, these
OTOH, As @sfackler said, if we change the methods to take the iterator by value, we can still reproduce the current behavior using the
Prints
IMO, I think the methods should take the iterator by value because depleted iterators are useless. Also, I think idiomatic code never calls these methods twice on the same iterator. cc @aturon This should get sorted out as part of |
Taking the iterator by value for the non-adaptor methods is strictly less powerful than taking it by reference. It's not always possible to move a value. |
Could you provide an example? (I thought this may be caused by the boxed closures, but the things I tried with a move collect worked) |
You can't move from |
Also, any method using by-value won't be available on a trait object. The adaptors use |
Sounds reasonable
Can't really comment on traits objects, because I've hardly used dynamic dispatch. But I recall some discussion about this issue. If making these Are there other methods that can take iterators by value but are taking them by reference? If not, then maybe this can be closed (?). |
I think it can be closed. It doesn't really apply to the methods on iterators themselves. |
To clarify a bit, the ergonomic tradeoff is different for adapter methods, because they will do the mutable borrowing automatically. The issue was talking about functions that take iterators as arguments; in that situation, you don't get automatic borrowing. |
Revert rust-lang#12947, trigger workspace switches on all structure changes again Closes rust-lang/rust-analyzer#13029
Functions that exhaust iterators should take them by value, even if they don't require ownership.
Example:
This optimizes the ergonomy of using them for the common case where you don't need access to the iterator anymore after it is empty, while still retaining the ability to do so with the
by_ref()
adapter:The text was updated successfully, but these errors were encountered: