在C语言环境下,使用setjmp库来实现简单的try-catch异常处理块

#include <setjmp.h>
#include <stdbool.h>
# define TRINT(EXP) jmp_buf EXP; typedef enum
# define TRY(EXP,JBRN) int JBRN=setjmp(EXP);if(!JBRN)
# define CATCH(JBRN,VAL) else if(JBRN==VAL)
# define THROW(EXP,VAL) (longjmp(EXP,VAL))
# define FINALLY() if(true)

setjmp.h是C语言的标准库,其中的setjmp函数可以保存当前的上下文,并且在调用longjmp函数时恢复上下文,实现跳转(可以跨函数跳转)

利用setjmp的特性,我们可以使用宏来实现简单的异常处理:

#include <stdio.h>
#include <stdbool.h>
#include <setjmp.h>
# define TRINT(EXP) jmp_buf EXP; typedef enum
# define TRY(EXP,JBRN) int JBRN=setjmp(EXP);if(!JBRN)
# define CATCH(JBRN,VAL) else if(JBRN==VAL)
# define THROW(EXP,VAL) (longjmp(EXP,VAL))
# define FINALLY() if(true)
TRINT(Exp) { ERR1, ERR2 } ExpVal;
int errfunc() {
    puts("[A]");
    THROW(Exp,ERR2);
    puts("[B]");
    return 0;
}
int main() {
    TRY(Exp,TryExp) {
        errfunc();
        puts("[C]");
    } CATCH(TryExp,ERR2) {
        puts("[D]");
    } FINALLY() {
        puts("[E]");
    }
    return 0;
}

让我们来分析这段代码。根据宏定义,我们可以将代码解析为下面这段代码:

#include <stdio.h>
#include <stdbool.h>
#include <setjmp.h>
jmp_buf Exp;
typedef enum { ERR1, ERR2 } ExpVal;
int errfunc() {
    puts("[A]");
    (longjmp(Exp,ERR2));
    puts("[B]");
    return 0;
}
int main() {
    int TryExp=setjmp(Exp);
    if(!TryExp) {
        errfunc();
        puts("[C]");
    } else if(TryExp==ERR2) {
        puts("[D]");
    }
    if(true) {
        puts("[E]");
    }
    return 0;
}

首先,在全局作用域里,我们声明了jmp_buf类型的变量Exp,它将会储存将要恢复的上下文

main函数中,我们首先定义了TryExp变量,赋值为setjmp(Exp)的结果,此时表达式结果为0

然后,在if-else if结构里,调用了errfunc函数,输出了"[A]",然后调用longjmp函数,它将会跳转到setjmp函数位置,同时恢复上下文,让setjmp函数返回ERR2,TryExp被赋值为ERR2

然后重新进入选择结构,来到第二个分支,输出了"[D]"

执行完毕后,来到if(true),执行FINALLY块中的代码,输出了"[E]"

所以最终输出为:

[A]
[D]
[E]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值