为什么对向量调用过滤器不会从向量中删除元素?

时间:2018-03-06 10:31:23

标签: rust higher-order-functions

我正在写一个小程序,找到一个马拉松冠军。 在我尝试为延迟了一段时间的跑步者过滤矢量之前,一切似乎都是合乎逻辑的。过滤器函数后向量保持不变,如果使用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);
}

2 个答案:

答案 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);
}