panic! 与不可恢复的错误

ch09-01-unrecoverable-errors-with-panic.md


commit a764530433720fe09ae2d97874c25341f8322573

突然有一天,糟糕的事情发生了,而你对此束手无策。对于这种情况,Rust 有 panic!宏。当执行这个宏时,程序会打印出一个错误信息,展开并清理栈数据,然后接着退出。出现这种情况的场景通常是检测到一些类型的 bug 而且程序员并不清楚该如何处理它。

Panic 中的栈展开与终止

当出现 panic! 时,程序默认会开始 展开unwinding),这意味着 Rust 会回溯栈并清理它遇到的每一个函数的数据,不过这个回溯并清理的过程有很多工作。另一种选择是直接 终止abort),这会不清理数据就退出程序。那么程序所使用的内存需要由操作系统来清理。如果你需要项目的最终二进制文件越小越好,可以由 panic 时展开切换为终止,通过在 Cargo.toml[profile] 部分增加 panic = 'abort'。例如,如果你想要在发布模式中 panic 时直接终止:

  1. [profile.release]
  2. panic = 'abort'

让我们在一个简单的程序中调用 panic!

文件名: src/main.rs

  1. fn main() {
  2. panic!("crash and burn");
  3. }

运行程序将会出现类似这样的输出:

  1. $ cargo run
  2. Compiling panic v0.1.0 (file:///projects/panic)
  3. Finished dev [unoptimized + debuginfo] target(s) in 0.25 secs
  4. Running `target/debug/panic`
  5. thread 'main' panicked at 'crash and burn', src/main.rs:2:4
  6. note: Run with `RUST_BACKTRACE=1` for a backtrace.

最后三行包含 panic! 造成的错误信息。第一行显示了 panic 提供的信息并指明了源码中 panic 出现的位置:src/main.rs:2:4 表明这是 src/main.rs 文件的第二行第四个字符。

在这个例子中,被指明的那一行是我们代码的一部分,而且查看这一行的话就会发现 panic! 宏的调用。在其他情况下,panic! 可能会出现在我们的代码调用的代码中。错误信息报告的文件名和行号可能指向别人代码中的 panic! 宏调用,而不是我们代码中最终导致 panic! 的那一行。可以使用 panic! 被调用的函数的 backtrace 来寻找(我们代码中出问题的地方)。下面我们会详细介绍 backtrace 是什么。

使用 panic! 的 backtrace

让我们来看看另一个因为我们代码中的 bug 引起的别的库中 panic! 的例子,而不是直接的宏调用。示例 9-1 有一些尝试通过索引访问 vector 中元素的例子:

文件名: src/main.rs

  1. fn main() {
  2. let v = vec![1, 2, 3];
  3. v[99];
  4. }

示例 9-1:尝试访问超越 vector 结尾的元素,这会造成 panic!

这里尝试访问 vector 的第一百个元素,不过它只有三个元素。这种情况下 Rust 会 panic。[] 应当返回一个元素,不过如果传递了一个无效索引,就没有可供 Rust 返回的正确的元素。

这种情况下其他像 C 这样语言会尝试直接提供所要求的值,即便这可能不是你期望的:你会得到任何对应 vector 中这个元素的内存位置的值,甚至是这些内存并不属于 vector 的情况。这被称为 缓冲区溢出buffer overread),并可能会导致安全漏洞,比如攻击者可以像这样操作索引来读取储存在数组后面不被允许的数据。

为了使程序远离这类漏洞,如果尝试读取一个索引不存在的元素,Rust 会停止执行并拒绝继续。尝试运行上面的程序会出现如下:

  1. $ cargo run
  2. Compiling panic v0.1.0 (file:///projects/panic)
  3. Finished dev [unoptimized + debuginfo] target(s) in 0.27 secs
  4. Running `target/debug/panic`
  5. thread 'main' panicked at 'index out of bounds: the len is 3 but the index is
  6. 99', /checkout/src/liballoc/vec.rs:1555:10
  7. note: Run with `RUST_BACKTRACE=1` for a backtrace.
  8. error: Process didn't exit successfully: `target/debug/panic` (exit code: 101)

这指向了一个不是我们编写的文件,libcollections/vec.rs。这是标准库中 Vec<T> 的实现。这是当对 vector v 使用 []libcollections/vec.rs 中会执行的代码,也是真正出现 panic! 的地方。

接下来的几行提醒我们可以设置 RUST_BACKTRACE 环境变量来得到一个 backtrace backtrace 是一个执行到目前位置所有被调用的函数的列表。Rust 的 backtrace 跟其他语言中的一样:阅读 backtrace 的关键是从头开始读直到发现你编写的文件。这就是问题的发源地。这一行往上是你的代码调用的代码;往下则是调用你的代码的代码。这些行可能包含核心 Rust 代码,标准库代码或用到的 crate 代码。让我们尝试获取一个 backtrace:示例 9-2 展示了与你看到类似的输出:

  1. $ RUST_BACKTRACE=1 cargo run
  2. Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs
  3. Running `target/debug/panic`
  4. thread 'main' panicked at 'index out of bounds: the len is 3 but the index is 99', /checkout/src/liballoc/vec.rs:1555:10
  5. stack backtrace:
  6. 0: std::sys::imp::backtrace::tracing::imp::unwind_backtrace
  7. at /checkout/src/libstd/sys/unix/backtrace/tracing/gcc_s.rs:49
  8. 1: std::sys_common::backtrace::_print
  9. at /checkout/src/libstd/sys_common/backtrace.rs:71
  10. 2: std::panicking::default_hook::{{closure}}
  11. at /checkout/src/libstd/sys_common/backtrace.rs:60
  12. at /checkout/src/libstd/panicking.rs:381
  13. 3: std::panicking::default_hook
  14. at /checkout/src/libstd/panicking.rs:397
  15. 4: std::panicking::rust_panic_with_hook
  16. at /checkout/src/libstd/panicking.rs:611
  17. 5: std::panicking::begin_panic
  18. at /checkout/src/libstd/panicking.rs:572
  19. 6: std::panicking::begin_panic_fmt
  20. at /checkout/src/libstd/panicking.rs:522
  21. 7: rust_begin_unwind
  22. at /checkout/src/libstd/panicking.rs:498
  23. 8: core::panicking::panic_fmt
  24. at /checkout/src/libcore/panicking.rs:71
  25. 9: core::panicking::panic_bounds_check
  26. at /checkout/src/libcore/panicking.rs:58
  27. 10: <alloc::vec::Vec<T> as core::ops::index::Index<usize>>::index
  28. at /checkout/src/liballoc/vec.rs:1555
  29. 11: panic::main
  30. at src/main.rs:4
  31. 12: __rust_maybe_catch_panic
  32. at /checkout/src/libpanic_unwind/lib.rs:99
  33. 13: std::rt::lang_start
  34. at /checkout/src/libstd/panicking.rs:459
  35. at /checkout/src/libstd/panic.rs:361
  36. at /checkout/src/libstd/rt.rs:61
  37. 14: main
  38. 15: __libc_start_main
  39. 16: <unknown>

示例 9-2:当设置 RUST_BACKTRACE 环境变量时 panic! 调用所生成的 backtrace 信息

这里有大量的输出!你实际看到的输出可能因不同的操作系统和 Rust 版本而有所不同。为了获取带有这些信息的 backtrace,必须启用 debug 标识。当不使用 —release 参数运行 cargo build 或 cargo run 时 debug 标识会默认启用,这里便是如此。

示例 9-2 的输出中,backtrace 的 11 行指向了我们项目中造成问题的行:src/main.rs 的第 4 行。如果你不希望程序 panic,第一个提到我们编写的代码行的位置是你应该开始调查的,以便查明是什么值如何在这个地方引起了 panic。在上面的例子中,我们故意编写会 panic 的代码来演示如何使用 backtrace,修复这个 panic 的方法就是不要尝试在一个只包含三个项的 vector 中请求索引是 100 的元素。当将来你的代码出现了 panic,你需要搞清楚在这特定的场景下代码中执行了什么操作和什么值导致了 panic,以及应当如何处理才能避免这个问题。

本章的后面会再次回到 panic! 并讲到何时应该及何时不应该使用这个方式。接下来,我们来看看如何使用 Result 来从错误中恢复。