6

我正在使用此正则表达式处理格式为“1s:1d”、“100:5000”等的简单字符串:

let retention_matcher = regex::Regex::new({r"^(\d+)([smhdy])?:(\d+)([smhdy])?$"}).unwrap();

我知道这个正则表达式应该只匹配一次,所以我想运行正则表达式进行捕获并检查捕获的数量。

    let iter = retention_matcher.captures_iter(ts);
    let count = iter.count();
    println!("iter.count(): {}", count);

    let _ : Vec<Option<(u64,u64)>> = iter.map(|regex_match| {
        let retval = retention_spec_to_pair(regex_match);
        println!("precision_opt: {:?}", retval);
        retval
    }).collect();

问题是该count()方法移动了iter,我不能再使用它了。

src/bin/whisper.rs:147:42: 147:46 error: use of moved value: `iter`
src/bin/whisper.rs:147         let _ : Vec<Option<(u64,u64)>> = iter.map(|regex_match| {
                                                                ^~~~
src/bin/whisper.rs:144:21: 144:25 note: `iter` moved here because it has type `regex::re::FindCaptures<'_, '_>`, which is non-copyable
src/bin/whisper.rs:144         let count = iter.count();

这对我来说没有意义。该方法是否应该count只返回一个可复制的usize值而不是移动iter?我该如何解决这个问题?

4

1 回答 1

7

我怀疑你正在考虑iter作为一个抽象的捕获序列。将其视为代表抽象捕获序列中的位置更为准确。任何迭代器都知道要做的基本事情是前进到序列中的下一项。也就是说,您可以提前位置并获取序列中的下一个元素。

count移动iter是因为为了计算序列中有多少元素,它必须生成整个序列。这必然会通过遍历整个序列来修改迭代器。它移动是因为在调用之后count,迭代器真的不再有用了。根据定义,它必须超过序列的末尾!

可悲的是,它看起来不像有问题的迭代器类型 ( FindCaptures) is Clone,所以你不能只复制它。

解决方案是重组您的代码以调用count. 如果您想获取第一个元素并确保没有更多元素,最简单的模式是:

let mut iter = ...;
if let Some(regex_match) = iter.next() {
    // Ensure that there is no next element.
    assert_eq!(iter.next(), None);

    // process regex_match ...
}
于 2015-06-01T09:12:54.257 回答