我正在写一个小程序,找到一个马拉松冠军。
在我尝试为延迟了一段时间的跑步者过滤矢量之前,一切似乎都是合乎逻辑的。过滤器函数后向量保持不变,如果使用iter_mut()
,则表示类型错误。
fn main() {
let mut input_line = String::new();
std::io::stdin().read_line(&mut input_line);
let n = input_line.trim().parse::<u8>().unwrap();
let mut v = Vec::with_capacity(n as usize);
for _ in 0..n {
let mut input_line = String::new();
std::io::stdin().read_line(&mut input_line);
let separated = input_line.trim().split(":").collect::<Vec<_>>();
let hours = separated[0].parse::<u8>().unwrap();
let minutes = separated[1].parse::<u8>().unwrap();
let seconds = separated[2].parse::<u8>().unwrap();
v.push((hours, minutes, seconds));
}
//println!("{:?}", v);
filter_hours(&mut v);
filter_minutes(&mut v);
filter_seconds(&mut v);
println!("{:?}", v[0]);
println!("{:?}", v);
}
fn filter_hours(v: &mut Vec<(u8, u8, u8)>) {
let (mut minimum, _, _) = v[0];
for &i in v.iter() {
let (h, _, _) = i;
if h < minimum {
minimum = h;
}
}
v.iter().filter(|&&(h, _, _)| h == minimum);
}
fn filter_minutes(v: &mut Vec<(u8, u8, u8)>) {
let (_, mut minimum, _) = v[0];
for &i in v.iter() {
let (_, m, _) = i;
if m < minimum {
minimum = m;
}
}
v.iter().filter(|&&(_, m, _)| m == minimum);
}
fn filter_seconds(v: &mut Vec<(u8, u8, u8)>) {
let (_, _, mut minimum) = v[0];
for &i in v.iter() {
let (_, _, s) = i;
if s < minimum {
minimum = s;
}
}
v.iter().filter(|&&(_, _, s)| s == minimum);
}
答案 0 :(得分:3)
注意filter
在迭代器上运行,而不在向量上运行;它从迭代器中删除元素,而不是从向量中删除元素。做你想做的事情的一种方法是将filter
的结果收集到一个新的向量中并用它替换旧的向量:v = v.iter().filter(whatever).collect();
但是这将为新的向量分配空间,从而复制元素旧的矢量进入新的,然后释放旧的矢量。
有一个实验性的API drain_filter
,它允许您修改向量并删除匹配的元素。然而,由于它是实验性的,因此该API仅在夜间可用。
如果你想保持稳定的Rust并避免collect
的开销,你需要手动删除元素。这样的事情应该这样做(取自drain_filter
文档):
let mut i = 0;
while i != vec.len() {
if some_predicate(&mut vec[i]) {
let val = vec.remove(i);
// your code here
} else {
i += 1;
}
}
答案 1 :(得分:3)
迭代器不会改变原始数据结构中的项目数。相反,您想使用retain
:
fn filter_hours(v: &mut Vec<(u8, u8, u8)>) {
let min = v.iter().map(|&(h, _, _)| h).min().unwrap();
v.retain(|&(h, _, _)| h == min);
}