std::longjmp
来自cppreference.com
在标头 <csetjmp> 定义
|
||
void longjmp( std::jmp_buf env, int status ); |
(C++17 前) | |
[[noreturn]] void longjmp( std::jmp_buf env, int status ); |
(C++17 起) | |
加载先前的 setjmp 调用所保存的执行上下文 env。此函数不会返回。控制被转移到设置了 env 的宏 setjmp 的调用点。该 setjmp 随后返回作为 status 传递的值。
如果调用了 setjmp 的函数已退出,那么行为未定义(也就是说只允许沿调用栈向上长跳)。
C++ 的额外限制
在 C 的 longjmp 基础上,C++ 的 std::longjmp
的行为受到更多限制。
如果分别以 throw 和 catch 替换 std::longjmp
和 setjmp 会执行任何自动对象的非平凡析构函数,那么这种 std::longjmp
的行为未定义。
在协程中可以使用 co_await 运算符的地方调用 |
(C++20 起) |
参数
env | - | 指代 setjmp 所保存的函数执行状态的变量 |
status | - | 从 setjmp 返回的值。如果它等于 0,那么用 1 代替 |
返回值
(无)
注解
std::longjmp
是 C 中处理函数无法有意义返回处的错误条件的机制。C++ 通常为此目的使用异常处理。
示例
运行此代码
#include <array> #include <cmath> #include <csetjmp> #include <cstdlib> #include <format> #include <iostream> std::jmp_buf solver_error_handler; std::array<double, 2> solve_quadratic_equation(double a, double b, double c) { const double discriminant = b * b - 4.0 * a * c; if (discriminant < 0) std::longjmp(solver_error_handler, true); // 去往错误处理器 const double delta = std::sqrt(discriminant) / (2.0 * a); const double argmin = -b / (2.0 * a); return {argmin - delta, argmin + delta}; } void show_quadratic_equation_solution(double a, double b, double c) { std::cout << std::format("求解 {}x² + {}x + {} = 0...\n", a, b, c); auto [x_0, x_1] = solve_quadratic_equation(a, b, c); std::cout << std::format("x₁ = {}, x₂ = {}\n\n", x_0, x_1); } int main() { if (setjmp(solver_error_handler)) { // 求解器的错误处理器 std::cout << "无实数解\n"; return EXIT_FAILURE; } for (auto [a, b, c] : {std::array{1, -3, 2}, {2, -3, -2}, {1, 2, 3}}) show_quadratic_equation_solution(a, b, c); return EXIT_SUCCESS; }
输出:
求解 1x² + -3x + 2 = 0... x₁ = 1, x₂ = 2 求解 2x² + -3x + -2 = 0... x₁ = -0.5, x₂ = 2 求解 1x² + 2x + 3 = 0... 无实数解
缺陷报告
下列更改行为的缺陷报告追溯地应用于以前出版的 C++ 标准。
缺陷报告 | 应用于 | 出版时的行为 | 正确行为 |
---|---|---|---|
LWG 619 | C++98 | C++ 额外限制的描述措辞比较模糊 | 改进措辞 |
LWG 894 | C++98 | 分别以 throw 和 catch 替换 std::longjmp 和 setjmp 会销毁任何自动对象时的行为未定义 |
只有在这种情况下会执行任何自动对象 的非平凡析构函数时行为才会未定义 |
参阅
保存上下文 (宏函数) |