命名空间
变体
操作

std::longjmp

来自 cppreference.cn
< cpp‎ | utility‎ | program
 
 
 
 
定义于头文件 <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 还有更多限制行为。

如果将 std::longjmp 替换为 throw,并将 setjmp 替换为 catch 会为任何自动对象调用非平凡析构函数,则此 std::longjmp 的行为是未定义的。

如果在可以使用 co_await 运算符的协程中调用 std::longjmp,则行为是未定义的。

(自 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); // Go to error handler
 
    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("Solving {}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))
    {
        // Error handler for solver
        std::cout << "No real solution\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;
}

输出

Solving 1x² + -3x + 2 = 0...
x₁ = 1, x₂ = 2
 
Solving 2x² + -3x + -2 = 0...
x₁ = -0.5, x₂ = 2
 
Solving 1x² + 2x + 3 = 0...
No real solution

[编辑] 缺陷报告

以下行为更改缺陷报告已追溯应用于先前发布的 C++ 标准。

DR 应用于 已发布行为 正确行为
LWG 619 C++98 C++ 中的额外限制措辞含糊 改进了措辞
LWG 894 C++98 如果替换行为未定义
std::longjmpthrowsetjmp
catch 会销毁任何自动对象
行为仅在以下情况下未定义
如果为
任何自动对象调用非平凡析构函数

[编辑] 参见

保存上下文
(函数宏) [编辑]
C 文档 关于 longjmp