news 2026/4/2 8:47:52

C语言中嵌套捕获异常

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C语言中嵌套捕获异常

在C++中我们可以使用try/catch来捕获异常,但是C语言中没有这样的关键字,只有setjmplongjmp函数,它们的常规用法是先设置信号捕获函数,再调用setjmp,在信号捕获函数中调用longjmp,如果出现异常则会跳到setjmp后面:

#include<setjmp.h>#include<signal.h>#include<stdio.h>jmp_buf buf;voidsig(ints){switch(s){caseSIGSEGV:caseSIGINT:caseSIGFPE:(void)signal(s,sig);longjmp(buf,1);break;default:break;}}intmain(intargc,char*argv[]){(void)signal(SIGINT,sig);(void)signal(SIGFPE,sig);(void)signal(SIGSEGV,sig);if(setjmp(buf)==0){int*p=nullptr;*p=0;}else{printf("exception\n");}

我们可以使用宏来模拟try,catch

#defineTRYif(setjmp(buf)==0){#defineCATCH\}\else

这样就可以写成下面的形式了:

TRY{}CATCH{}

目前这种形式不支持嵌套捕获,比如:

TRY{TRY{}CATCH{}//如果这里出现异常,就不能捕获}CATCH{}

要支持捕获,也很简单,把jmp_buf设置成数组,再使用一个变量来记录深度,使用一次TRY则加1,CATCH时减1即可:

jmp_buf buf[256]={};uint8_tdepth=0;#defineTRYif(setjmp(buf[depth++])==0){#defineCATCH\--depth;\}\elsevoidsig(ints){switch(s){caseSIGSEGV:caseSIGINT:caseSIGFPE:(void)signal(s,sig);longjmp(buf[--depth],1);break;default:break;}}

测试代码:

#include<setjmp.h>#include<signal.h>#include<stdint.h>#include<stdio.h>jmp_buf buf[256]={};uint16_tdepth=0;#defineTRYif(setjmp(buf[depth++])==0){#defineCATCH\--depth;\}\elsevoidsegv(){int*v=nullptr;*v=1;printf("%p\n",v);}voiddiv0(){intv=0;printf("%d\n",1/v);}voidsig(ints){switch(s){caseSIGSEGV:caseSIGINT:caseSIGFPE:(void)signal(s,sig);longjmp(buf[--depth],1);break;default:break;}}voidtest(){TRY{TRY{segv();}CATCH{printf("SIGSEGV exception\n");}div0();}CATCH{printf("SIGFPE exception\n");}}intmain(intargc,char*argv[]){(void)signal(SIGINT,sig);(void)signal(SIGFPE,sig);(void)signal(SIGSEGV,sig);TRY{test();}CATCH{printf("test exception\n");}printf("ok\n");return0;}

如果是使用的VC编译器和调试器,在除0的地方会直接断下来,并不会继续执行,所以触发不了SIGFPE信号,这是由于SEH异常机制导致,如果想要实现与GCC一样的行为,即POSIX行为,则需要拦截SEH,手动触发SIGFPE信号:

#if_MSC_VER#include<windows.h>LONG WINAPIseh_handler(EXCEPTION_POINTERS*e){if(e->ExceptionRecord->ExceptionCode==EXCEPTION_INT_DIVIDE_BY_ZERO){raise(SIGFPE);returnEXCEPTION_CONTINUE_EXECUTION;}returnEXCEPTION_CONTINUE_SEARCH;}#endif

再调用:

#if_MSC_VERAddVectoredExceptionHandler(1,seh_handler);#endif

完整代码:

#include<setjmp.h>#include<signal.h>#include<stdint.h>#include<stdio.h>#if_MSC_VER#include<windows.h>LONG WINAPIseh_handler(EXCEPTION_POINTERS*e){if(e->ExceptionRecord->ExceptionCode==EXCEPTION_INT_DIVIDE_BY_ZERO){raise(SIGFPE);returnEXCEPTION_CONTINUE_EXECUTION;}returnEXCEPTION_CONTINUE_SEARCH;}#endifjmp_buf buf[256]={};uint16_tdepth=0;#defineTRYif(setjmp(buf[depth++])==0){#defineCATCH\--depth;\}\elsevoidsegv(){int*v=0;*v=1;printf("%p\n",v);}voiddiv0(){intv=0;printf("%d\n",1/v);}voidsig(ints){switch(s){caseSIGSEGV:caseSIGINT:caseSIGFPE:(void)signal(s,sig);longjmp(buf[--depth],1);break;default:break;}}voidtest(){TRY{TRY{segv();}CATCH{printf("SIGSEGV exception\n");}div0();}CATCH{printf("SIGFPE exception\n");}}intmain(intargc,char*argv[]){#if_MSC_VERAddVectoredExceptionHandler(1,seh_handler);#endif(void)signal(SIGINT,sig);(void)signal(SIGFPE,sig);(void)signal(SIGSEGV,sig);TRY{TRY{TRY{segv();}CATCH{printf("SIGSEGV exception\n");}div0();}CATCH{printf("SIGFPE exception\n");}test();}CATCH{printf("test exception\n");}printf("ok\n");return0;}

如果对你有帮助,欢迎点赞收藏!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/1 0:29:23

【计算机毕业设计案例】基于 Spring Boot+MySQL 的汽车租赁管理系统设计与实现基于MyBatis的在线车辆租赁信息管理系统的设计与实现(程序+文档+讲解+定制)

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

作者头像 李华
网站建设 2026/3/27 13:31:13

Word自动编号完美对齐技巧

方法一&#xff1a;精确调整列表缩进&#xff08;推荐&#xff09;创建或选中自动编号段落&#xff0c;通过修改底层格式实现永久对齐。打开“定义新编号格式”对话框&#xff1a; 在「开始」选项卡的「段落」组中&#xff0c;点击「编号」下拉箭头&#xff0c;选择「定义新编号…

作者头像 李华
网站建设 2026/3/27 21:29:11

Word题注编号间距调整4种方法

修改题注编号与文字间距的方法方法一&#xff1a;修改题注编号格式 在“引用”选项卡中点击“插入题注”&#xff0c;弹出对话框后点击“编号”按钮。在“题注编号”对话框中&#xff0c;直接在“题注”输入框的编号后添加空格或符号&#xff08;如图 1.1 或图 1.1 -&#xff0…

作者头像 李华
网站建设 2026/3/31 2:03:12

从遮蔽到表达:当地板行业迎来它的“新质生产力”

一旦需求变化&#xff0c;最先陷入被动的&#xff0c;就是曾经的王者。与其内卷&#xff0c;不如开启一场“大迁徙”。 ——知名战略咨询顾问 刘润 七千年前&#xff0c;河姆渡人执石斧斫木&#xff0c;以榫卯之巧拼接&#xff0c;以粗糙的木痕&#xff0c;铺就了人类对地面…

作者头像 李华
网站建设 2026/3/15 0:29:48

Python+django基于微信小程序的医院医疗设备报修管理系统设计-

文章目录 系统概述技术架构核心功能创新点应用价值 系统设计与实现的思路主要技术与实现手段源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01; 系统概述 该系统基于PythonDjango框架与微信小程序开发&#xff0c;旨在实现医院医疗设备的数…

作者头像 李华