6

Я хочу, чтобы написать что-то вроде этого:Невозможно использовать или отбрасывать конструктор как Fn

use std::{iter, ops}; 

struct Idx(usize); 

fn get_inds() -> iter::Zip<iter::Map<ops::RangeFrom<usize>, fn(usize) -> Idx>, ops::RangeFrom<usize>> { 
    (0..).map(Idx).zip(0..) 
} 

Однако это не удается скомпилировать:

error: mismatched types [--explain E0308] 
--> src/main.rs:6:9 
6 |>   (0..).map(Idx).zip(0..) 
    |>   ^^^^^^^^^^^^^^^^^^^^^^^ expected fn pointer, found fn item 
note: expected type `std::iter::Zip<std::iter::Map<std::ops::RangeFrom<usize>, fn(usize) -> Idx>, std::ops::RangeFrom<usize>>` 
note: found type `std::iter::Zip<std::iter::Map<std::ops::RangeFrom<usize>, fn(usize) -> Idx {Idx::{{constructor}}}>, std::ops::RangeFrom<_>>` 

Затем я бросил функцию:

fn get_inds() -> iter::Zip<iter::Map<ops::RangeFrom<usize>, fn(usize) -> Idx>, ops::RangeFrom<usize>> { 
    (0..).map(Idx as fn(usize) -> Idx).zip(0..) 
} 

который генерирует следующее предупреждение о компиляторе. Почему я получаю предупреждение? Каков правильный способ использования конструктора?

warning: can't cast this type, #[warn(const_err)] on by default 
--> src/main.rs:6:19 
6 |>   (0..).map(Idx as fn(usize) -> Idx).zip(0..) 
    |>     ^^^^^^^^^^^^^^^^^^^^^^^ 
+0

@Ruud 'fn (A) -> B desugars to std :: ops :: Fn <(A,), Output = B>' - нет, это не так. Функция * реализует * черты 'Fn *', но они не совпадают. Так же, как 'i32' * реализует *' Добавить', но не добавляется в 'Добавить'. – Shepmaster

ответ

2

Это поддельное предупреждение, вызванное недавними изменениями в постоянном оценщике компилятора. См. #33452, #33291 и т. Д. Я бы не стал беспокоиться об этом, но стоит отправить ошибку в репозиторий Rust, чтобы он мог быть исправлен.