反转字符串

时间:2017-11-23 12:51:25

标签: rust concatenation reverse substr

我正在通过代码练习练习,并最终得到一些混乱的代码,在Python中,相对简单的字符串切片和连接:

s = "the quick (brown (fox) jumps) over"
(f, t) = (18, 21)
ss = s[:f] + s[t-1:f-1:-1] + s[t:]

在Rust:

let s = String::from("the quick (brown (fox) jumps) over");
let sc: Vec<_> = s.chars().collect();
let (f, t) = (18, 21); // result of searching through sc
let mut new_s: Vec<_> = sc[..f].into_iter().collect();
new_s.extend(&sc[f..t].into_iter().rev().collect::<Vec<_>>());
new_s.extend(&sc[t..].into_iter().collect::<Vec<_>>());
let mut ss = String::with_capacity(new_s.len());
for c in new_s.iter() {ss.push(**c);}

主要是通过添加修复编译错误的东西,我最终得到的复杂性似乎无法减少。什么是适当的&#39;在Rust中这样做的方法?

我没有使用replace()做类似回答的事情,因为这是搜索字符串并反转匹配括号内部分的问题的简化示例。

2 个答案:

答案 0 :(得分:5)

这应该只为你的5分配两次,我不确定是否有办法摆脱中间分配:

<p>

请注意,UTF-8字符串操作比Python代码似乎暗示要复杂得多。例如,在存在组合字符(playground)的情况下,两种解决方案都无法达到预期效果。

以下摆脱了中间分配:

let ss: String = s.chars()
                  .take(f)
                  .chain(s.chars().skip(f).take(t-f).collect::<Vec<_>>().into_iter().rev())
                  .chain(s.chars().skip(t))
                  .collect();

对于Unicode,我们需要使用let ss: String = s.chars() .take(f) .chain(s.chars().rev().skip(s.chars().count()-t).take(t-f)) .chain(s.chars().skip(t)) .collect(); 而不是s.chars().count(),因为s.len()是代码单元,我们需要代码点的长度。

这更容易理解,但仅适用于ASCII或s.len()f是代码单位的索引:

t

答案 1 :(得分:1)

假设子串出现一次(或者想要反转所有子串):

fn main() {
    let s = String::from("the quick (brown (fox) jumps) over");
    let to_rev = &"fox";
    let ss = s.replace(to_rev, &to_rev.chars().rev().collect::<String>());

    println!("{}", ss); // the quick (brown (xof) jumps) over
}