我正在尝试读取Rust中的文件。我不明白的是:当BufferedReader
在EOF时,它实际上给了一个Err(IoError {kind: EndOfFile})
,我不知道如何匹配它。
loop {
match file.read_line() {
Ok(line) => {
// Do stuff
},
// Err(EndOfFile) => is want I want here so I can call "break"
Err(_) => {
panic!("Unexpected error reading file.");
}
}
}
如何明确匹配EndOfFile
枚举变体?
如何明确匹配EndOfFile枚举变体?
您可以使用以下模式匹配它而无需额外的嵌套匹配:
loop {
match file.read_line() {
Ok(line) => {
// Do stuff
},
Err(IoError { kind: EndOfFile, .. }) =>
break,
Err(_) => {
panic!("Unexpected error reading file.");
}
}
}
特别是对于线路迭代,具有缓冲区(lines
)的http://doc.rust-lang.org/std/io/trait.BufferPrelude.html#tymethod.lines函数。
在你的情况下,你将循环遍历线,一旦达到EOF,循环将自动中止而无需您的干预。
for line in file.lines() {
match line {
Ok(line) => {
// do stuff
},
Err(_) => {
println!("Unexpected error reading file.")
}
}
}
或者,如果您的函数返回兼容的Result
,您可以使用try!减少噪音的宏观:
fn myfun(file: File) -> IoResult<()> {
for line in file.lines() {
let line = try!(line);
// do stuff
}
}
看起来生锈后1.0已经将IoError
改为std::io::Error
。此外,kind
现在隐藏在内部。我不得不使用guard来处理类似的问题(WouldBlock
而不是EndOfFile
)。
EG
match some_method_that_might_block() {
Ok(_) => {
debug!("worked!");
},
Err(ref e @ IoError { kind: std::io::ErrorKind::WouldBlock, .. }) if e.kind() == std::io::ErrorKind::WouldBlock => {
// ignore WouldBlock
},
Err(e) => { debug!("error={:?}", e); }
};
对于在现代版本的锈色贴1.0上阅读此内容的任何人,the docs现在规定,在尝试从中读取时到达文件的末尾绝对不是错误,应该用Ok(0)
表示(很像其他编程语言/框架)而不是一个硬错误。
因为你可以直接匹配读取的字节数(不像std::io::error
成员隐藏的ErrorKind
,不能直接匹配),这使得处理它非常容易:
let mut buf: Vec<u8> = Vec::new();
loop {
let bytes_read = match file.read(&mut buf) {
Ok(0) => break, // end-of-file
Ok(n) => n,
Err(e) => // handle error however you see fit,
};