我正在尝试使用闭包来调用一堆特征向量上的函数。这里最重要的代码行是第一行,我们说的是query
必须是什么类型,最后一行是我们尝试给出一个闭包。
fn stat_query(dataset: &Vec<Vec<i32>>, query: Fn(&Vec<i32>) -> f64) -> f64 {
// takes in a dataset of feature vectors and a query,
// returns the average of the query on the dataset
if dataset.len() == 0 {
0.
} else {
dataset.iter().map(|ref fv| query(&fv)).sum() / (dataset.len() as f64)
}
}
fn main() {
// build dataset
let fv1: Vec<i32> = vec![1, 1, 1, 1, 1];
let fv2: Vec<i32> = vec![1, 0, 1, 0, 1];
let my_dataset = vec![fv1, fv2];
// query checks whether sum of features is greater than a threshold
fn my_query(ref fv: &Vec<i32>, threshold: i32) -> f64 {
if fv.iter().sum() > threshold {
1.
} else {
0.
}
}
// run query on dataset with threshold 3
println!("{}", stat_query(&my_dataset, |ref fv| my_query(fv, 3)));
}
当我运行时,我收到错误:
error[E0277]: the trait bound `for<'r> std::ops::Fn(&'r std::vec::Vec<i32>) -> f64 + 'static: std::marker::Sized` is not satisfied
--> src/main.rs:1:40
|
1 | fn stat_query(dataset: &Vec<Vec<i32>>, query: Fn(&Vec<i32>) -> f64) -> f64 {
| ^^^^^ `for<'r> std::ops::Fn(&'r std::vec::Vec<i32>) -> f64 + 'static` does not have a constant size known at compile-time
|
= help: the trait `std::marker::Sized` is not implemented for `for<'r> std::ops::Fn(&'r std::vec::Vec<i32>) -> f64 + 'static`
= note: all local variables must have a statically known size
error[E0308]: mismatched types
--> src/main.rs:27:44
|
27 | println!("{}", stat_query(&my_dataset, |ref fv| my_query(fv, 3)));
| ^^^^^^^^^^^^^^^^^^^^^^^^ expected trait std::ops::Fn, found closure
|
= note: expected type `for<'r> std::ops::Fn(&'r std::vec::Vec<i32>) -> f64 + 'static`
found type `[closure@src/main.rs:27:44: 27:68]`
我认为闭包implemented Fn
(或FnOnce
或FnMut
)?如果期望Fn
,提供闭包是否无效?我想我需要在第一行中指定有关Fn
的其他内容吗?但是什么?
我假设我错误地指定了我的第一个函数的query: Fn(&Vec<i32>) -> f64
参数的类型,或者写错了|ref fv| my_query(fv, 3)
。
我读过Passing closure to trait method: expected type parameter, found closure,但这似乎更多的是关于传递一个闭包,这不是函数接受的唯一一种东西。
答案 0 :(得分:3)
第二个错误是更重要的错误:
Fn(&Vec<i32>) -> f64
简而言之:fn stat_query<F>(dataset: &Vec<Vec<i32>>, query: F) -> f64
where
F: Fn(&Vec<i32>) -> f64,
{
// takes in a dataset of feature vectors and a query,
// returns the average of the query on the dataset
if dataset.len() == 0 {
0.
} else {
dataset.iter().map(|ref fv| query(&fv)).sum::<f64>() / (dataset.len() as f64)
}
}
fn main() {
// build dataset
let fv1: Vec<i32> = vec![1, 1, 1, 1, 1];
let fv2: Vec<i32> = vec![1, 0, 1, 0, 1];
let my_dataset = vec![fv1, fv2];
// query checks whether sum of features is greater than a threshold
fn my_query(ref fv: &Vec<i32>, threshold: i32) -> f64 {
if fv.iter().sum::<i32>() > threshold {
1.
} else {
0.
}
}
// run query on dataset with threshold 3
println!("{}", stat_query(&my_dataset, |ref fv| my_query(fv, 3)));
}
不是大小类型,不能按值传递。您只能传递trait objects(引用或框)或实现特征的(大小)类型的值。在这种情况下,你可能想要后者:
$(this).length