-
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
compiler fails to suggest move
closure
#61909
Comments
Simpler reproduction: https://rust.godbolt.org/z/Ex3HIu |
It's interesting to compare to E0373: |
I ran into this issue today while working on a small project I'm using to learn Rust. I was quite stumped by the error I was seeing until some googling led me to this issue. The code in my case looked like this: Click to Expand// Grid is a thin wrapper around a HashMap<(i64, i64), Cell> representing a 2D grid of values.
// Cell here is just an enum, not std::cell::Cell.
fn get_intersections<'a>(grid: &'a Grid<Cell>) -> impl Iterator<Item = &'a Coord> {
grid.iter()
// Filter to nonempty cells.
.filter_map(|(coord, cell)| {
if *cell != Cell::Empty {
Some(coord)
} else {
None
}
})
// Intersections are nonempty locations with at least 3 nonempty neighbors.
.filter(|coord| { // This needs to be a `move` closure to capture `grid`.
let num_adjacent = grid
.neighbors(coord)
.filter(|&(_, cell)| cell != Cell::Empty)
.count();
num_adjacent >= 3
})
} and the error I get from rustc 1.40 is: Click to Expand
As a relatively new rust user, I was confused because I thought this error message was telling me that my @nikomatsakis do you have a sense of how hard it might be to try to fix or improve this error? I'd be interested in taking a crack at it if you think it'd be a reasonable starting point. |
I was wondering recently if the current reference-capturing behavior of Consider this simple code: fn f<'a>(v: &'a [i32]) -> impl Iterator<Item = i32> + 'a {
v.iter().map(|e| if v.len() > 10 { e * 2 } else { e * 4 })
}
fn main() {
let v = vec![1, 2, 3];
for e in f(&v) {
dbg!(e);
}
} It won't compile because we're essentially capturing the pointer to However, I'd argue that the majority of users would expect that code to work, since it's very uncommon, I believe, to intentionally borrow a local reference. Even if we change reference-capturing behavior of fn f<'a>(v: &'a [i32]) -> impl Iterator<Item = i32> + 'a {
let r = &v;
v.iter().map(|e| if r.len() > 10 { e * 2 } else { e * 4 })
} So, I think current |
@yuyoyuppe this can't be lightly changed in a backwards compatible fashion. We have to decide the mode of capture before we learn about the error that results from this code. Furthermore, there are cases where the choice to capture "in place" or "by move" results in user-visible distinctions in terms of what drops and what doesn't, so it's best for the rules there to be relatively predictable. |
Even simpler test case: fn main() {
let _f = {
let i = 1;
|| i
};
}
This one did also not have a suggestion for |
Triage: the original report is fixed
as well as the report in one of the comments
but the last repro isn't
We should also try to group together the errors in the first case, for multiple invalid captures in the same closure and emit a single error. Edit: Looking at it, the final repro is covered by #58497, so I'm going to close this one. |
This example gives a fairly opaque error message (playground):
We currently get:
But as of 1.34.0 we used to get:
(There are also errors about
key
that are similar.)The text was updated successfully, but these errors were encountered: