setjmp和longjmp简介

setjmp和longjmp简介

1setjmp和longjmp简介

  与刺激的abort函数和exit函数相比,goto语句看起来是处理异常的更可行方案。但是goto是本地的,它只能跳到所在函数内部的标号上,而不能将控制权转移到所在程序的任意地点(当然,除非你的所有代码都在main体中)。
为了解决这个限制,C函数库提供了setjmp函数和longjmp函数,它们分别承担非局部标号和goto作用。头文件<setjmp.h>申明了这些函数及同时所需的jmp_buf数据类型。
  原理非常简单:
  1. setjmp(j)设置“jump”点,用正确的程序上下文填充jmp_buf对象j。这个上下文包括程序存放位置、栈和框架指针,其它重要的寄存器和内存数据。当初始化完jump的上下文,setjmp函数返回0值。
  2. 2.以后调用longjmp(j,r)的效果就是一个非局部的goto或“长跳转”到由j描述的上下文处(也就是到那原来设置j的setjmp函数处)。当作为长跳转的目标而被调用时,setjmp()返回r或1(如果r设为0的话)。(记住,setjmp函数不能在这种情况时返回0。)
  通过有两类返回值,setjmp函数让你知道它正在被怎么使用。当设置j时,setjmp函数如你期望地执行;但当作为长跳转的目标时,setjmp函数就从外面“唤醒”它的上下文。你可以用longjmp函数来终止异常,用setjmp函数标记相应的异常处理程序。常见的使用方法如下所示。

#include <setjmp.h>
#include <stdio.h>

jmp_buf j;

void raise_exception(void)
{
printf("exception raised\n");
longjmp(j, 3); /* jump to exception handler case 3 */
printf("this line should never appear\n");
}

int main(void)
{
switch (setjmp(j)) {
case 0:
printf("‘‘setjmp‘‘ is initializing ‘‘j‘‘\n");
raise_exception();
printf("this line should never appear\n");
case 1:
printf("Case 1\n");break;
case 2:
printf("Case 2\n");break;
case 3:
printf("Case 3\n");break;
default:
break;
}
return 0;
}

  最终的输出结果如下所示。

‘‘setjmp‘‘ is initializing ‘‘j‘‘
exception raised
Case 3

2SylixOS下实现

  SylixOS未使用C库中的setjmp与longjmp函数,而是重新在各个架构中实现。如MIPS架构便是在/libsyllixos/SylixOS/arch/mips/common/mipsLibAsm.S文件中实现这两个函数,其代码如下所示。

;/**************************************************************************
;  sigsetjmp (参数为 jmp_buf, mask_saved)
;**************************************************************************/

FUNC_DEF(sigsetjmp)
    CALL_SETUP  __sigsetjmpSetup

    SAVE_NON_VOLATILE_REG

    JR          RA
    MOV         V0 , ZERO
    FUNC_END(sigsetjmp)
;/**************************************************************************
;  longjmp (参数为 jmp_buf, retval)
;**************************************************************************/

FUNC_DEF(longjmp)
CALL_SETUP  __longjmpSetup

RESTORE_NON_VOLATILE_REG

MOV         V0 , A1
BNE         V0 , ZERO , longjmpnotzero
NOP
LI          V0 , 1
LINE_LABEL(longjmpnotzero)
JR          RA
NOP
FUNC_END(longjmp)

3 参考资料

  《百度百科》

原文地址:http://blog.51cto.com/12557713/2153046

时间: 2024-12-16 07:30:54

setjmp和longjmp简介的相关文章

setjmp()和longjmp()

1.语言没有异常处理机制,可以使用setjmp和longjmp进行模拟,另外,goto语句不能在函数间跳转,可以使用setjmp和longjmp完成函数间的跳转. 使用setjmp()函数需要包含头文件<setjmp.h> setjmp()函数用于建立本地的jmp_buf缓冲区并初始化,用于将来跳转到这个地方. longjmp()函数恢复jmp_buf变量所保存的信息,longjmp()之后返回setjmp()处执行. 1.1使用setjmp处理异常的例子: # include <std

学习之&quot;setjmp和longjmp函数&quot;

Linux学习之"setjmp和longjmp函数" 转贴,原文地址:http://www.cnblogs.com/lq0729/archive/2011/10/23/2222117.html nsetjmp和longjmp函数实现函数之间的跳转(需包含头文件"setjmp.h"): 函数原型:int setjmp(jmp_buf env);   void longjmp(jmp_buf env, int val); setjmp函数用于设置跳转的目的位置,long

非本地跳转之setjmp与longjmp

非本地跳转(unlocal jump)是与本地跳转相对应的一个概念. 本地跳转主要指的是类似于goto语句的一系列应用,当设置了标志之后,可以跳到所在函数内部的标号上.然而,本地跳转不能将控制权转移到所在程序的任意地点,不能跨越函数,因此也就有了非本地跳转. C语言里面提供了setjmp和longjmp函数来进行跨越函数之间的控制权的跳转,从而称之为非本地跳转. #include <setjmp.h> int setjmp(jmp_buf env); 该函数主要用来保存当前执行状态,作为后续跳

C中的setjmp与longjmp

setjmp与longjmp是属于C语言中的,当然,C++也会有这两个函数了.他们的原型如下: int setjmp( jmp_buf env ); 作用:第一次调佣时,将寄存器的当前状态信息全部存入到env中,并返回0.如果在某处调用了longjmp(env,x),且x!=0,则setjmp的返回值将设为x.而若x==0,则setjmp返回1. void longjmp( jmp_buf env,int value ); 作用:重新存储当前寄存器的状态信息,并将setjmp的返回值设为valu

setjmp和longjmp

此文是学习 C专家编程 中的笔记. setjmp和longjmp是C语言所独有的,它们部分弥补了C语言有限的转移能力. 函数说明(来自wiki百科): int setjmp(jmp_buf env) 建立本地的jmp_buf缓冲区并且初始化,用于将来跳转回此处.这个子程序保存程序的调用环境于env参数所指的缓冲区,env将被longjmp使用.如果是从setjmp直接调用返回,setjmp返回值为0.如果是从longjmp恢复的程序调用环境返回,setjmp返回非零值. void longjmp

【C语言天天练(五)】setjmp和longjmp

setjmp和longjmp组合可以实现跳转,与goto语句有相似的地方.但有以下不同: 1.用longjmp只能跳回到曾经到过的地方.在执行setjmp的地方仍留有一个过程活动记录.从这个角度看,longjmp更像是"从何处来"而不是"往何处去".longjmp接收一个额外的整型参数并返回它的值,这可以知道是由longjmp转移到这里的还是从上一条语句执行后自然而然来到这里的. 2.goto语句不能跳出C语言当前的函数,而longjmp可以跳的更远,可以跳出函数,

C语言中利用setjmp和longjmp做异常处理

错误处理是任何语言都需要解决的问题,只有不能保证100%的正确运行,就需要有处理错误的机制.异常处理就是其中的一种错误处理方式. 1 过程活动记录(Active Record) C语言中每当有一个函数调用时,就会在堆栈(Stack)上准备一个被称为AR的结构,抛开具体编译器实现细节的不同,这个AR基本结构如下所示. 每当遇到一次函数调用的语句,C编译器都会产生出汇编代码来在堆栈上分配这个AR.例如下面的C代码: void a(int i) { if(i==0){ i = 1; } else {

setjmp()与longjmp()

setjmp()与longjmp() 和 函数与return有点类似 setjmp() : 直接调用时返回0,通过longjmp调用,返回longjmp指定的数值 longjmp():调用longjmp()运行的指针会跳到 例子 #include <stdio.h> #include <setjmp.h> static jmp_buf buf; void second(void) { printf("second\n"); // 打印 longjmp(buf,1

Linux系统编程_7_进程环境之setjmp和longjmp函数

大家都知道C语言中goto关键字可以用来跳转,但你知道它的跳转范围是什么吗? goto语句只能在当前函数内不跳转,不能实现跨函数跳转: 为实现这一目的,Linux中引入了setjmp和longjmp,这两个函数对于处理发生深层嵌套函数调用中的出错情况非常有用. 函数声明: #include <setjmp.h> int setjmp(jmp_buf env); //env是jmp_buf类型,一般定义为全局变量 void longjmp(jmp_buf env, int val);   //v