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

a boxed iterator can't use methods in the IteratorExt trait #21361

Closed
Kroisse opened this issue Jan 18, 2015 · 3 comments · Fixed by #21392
Closed

a boxed iterator can't use methods in the IteratorExt trait #21361

Kroisse opened this issue Jan 18, 2015 · 3 comments · Fixed by #21392
Labels
A-trait-system Area: Trait system

Comments

@Kroisse
Copy link
Contributor

Kroisse commented Jan 18, 2015

Example

let v = vec![1, 2, 3];
let boxed = Box::new(v.iter()) as Box<Iterator<Item=i32>>;
boxed.max()  // IteratorExt::max(*boxed) also fails

This code will be failed to compile because boxed holds a trait object that is unsized, and IteratorExt requires the Sized kind so IteratorExt can only be implemented for sized types. Like IteratorExt::map(), many methods in IteratorExt consume self, so requiring a sized type is necessary.

Adding impl Iterator for Box<Iterator> (like Reader for Box<Reader>) might solve this problem, but I'm worrying about the confliction with autoderef.

@japaric
Copy link
Member

japaric commented Jan 18, 2015

I think a possible solution would be impl Iterator for &mut Iterator, that way boxed.max() would autoderef boxed from Box<Iterator> into &mut Iterator (because Box implements DerefMut<Target=T>) and then since &mut Iterator implements the Iterator trait, it would also implement IteratorExt. Let me see if that works.

@japaric
Copy link
Member

japaric commented Jan 18, 2015

The idea works with the old Iterator<T> trait:

#![no_implicit_prelude]

use std::boxed::Box;
use std::marker::Sized;
use std::option::Option;

trait Iterator<T> {
    fn next(&mut self) -> Option<T>;
}

impl<'a, T> Iterator<T> for &'a mut (Iterator<T> + 'a) {
    fn next(&mut self) -> Option<T> {
        loop {}
    }
}

trait IteratorExt<T>: Iterator<T> + Sized {
    fn last(self) -> Option<T> {
        loop {}
    }
}

impl<I, T> IteratorExt<T> for I where I: Iterator<T> {}

fn test(mut it: Box<Iterator<u8> + 'static>) {
    it.last();  // <-- it works
}

fn main() {}

But I'm not quite sure how to impl<'a> Iterator for &'a mut (Iterator + 'a) with associated types. This gets rejected:

impl<'a, T> Iterator for &'a mut (Iterator<Item=T> + 'a) {
//~^ error: the type parameter `T` is not constrained by the impl trait, self type, or predicates
    type Item = T;
}

@japaric
Copy link
Member

japaric commented Jan 18, 2015

I think the last impl should be accepted, but it's not because of a bug, so I have filed #21363.

japaric pushed a commit to japaric/rust that referenced this issue Jan 19, 2015
alexcrichton added a commit to alexcrichton/rust that referenced this issue Jan 21, 2015
closes rust-lang#20953
closes rust-lang#21361

---

In the future, we will likely derive these `impl`s via syntax extensions or using compiler magic (see rust-lang#20617). For the time being we can use these manual `impl`s.

r? @aturon
cc @BurntSushi @Kroisse
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-trait-system Area: Trait system
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants