Мне нравится использовать итераторы в Rust, и я думаю, что это идеальный вариант использования:
#[derive(Debug, PartialEq)]
enum MyError {
EscapeAtEndOfString,
InvalidEscapedChar(char),
}
struct InterpretEscapedString<'a> {
s: std::str::Chars<'a>,
}
impl<'a> Iterator for InterpretEscapedString<'a> {
type Item = Result<char, MyError>;
fn next(&mut self) -> Option<Self::Item> {
self.s.next().map(|c| match c {
'\\' => match self.s.next() {
None => Err(MyError::EscapeAtEndOfString),
Some('n') => Ok('\n'),
Some('\\') => Ok('\\'),
// etc.
Some(c) => Err(MyError::InvalidEscapedChar(c)),
},
c => Ok(c),
})
}
}
fn interpret_escaped_string(s: &str) -> Result<String, MyError> {
(InterpretEscapedString { s: s.chars() }).collect()
}
fn main() {
assert_eq!(interpret_escaped_string(r#""#), Ok("".into()));
assert_eq!(interpret_escaped_string(r#"a"#), Ok("a".into()));
assert_eq!(interpret_escaped_string(r#"\"#), Err(MyError::EscapeAtEndOfString));
assert_eq!(interpret_escaped_string(r#"\\"#), Ok("\\".into()));
assert_eq!(interpret_escaped_string(r#"a\n"#), Ok("a\n".into()));
assert_eq!(interpret_escaped_string(r#"a\."#), Err(MyError::InvalidEscapedChar('.')));
}
Более полная реализация такого модуля на игровой площадке .