正则表达式捕获迭代器方法移动迭代器

xrl*_*xrl 6 regex iterator rust borrow-checker

我用这个正则表达式处理格式为"1s:1d","100:5000"等的简单字符串:

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

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

    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();
Run Code Online (Sandbox Code Playgroud)

问题是该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();
Run Code Online (Sandbox Code Playgroud)

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

DK.*_*DK. 7

我怀疑你是iter一个抽象的捕获序列.将其视为表示抽象的捕获序列中的位置更准确.任何迭代器知道要做的基本事情是前进到序列中的下一个项目; 也就是说,您可以提升位置并获得序列中的下一个元素.

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

可悲的是,它看起来不像问题(FindCaptures)中的迭代器类型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 ...
}
Run Code Online (Sandbox Code Playgroud)