#include <setjmp.h> |
Returns: 0 if called directly, nonzero if returning from a call to longjmp.
|
void longjmp(jmp_buf env, int val); |
這兩個函數(shù)都要包含頭文件setjmp.h。而且它們在處理出現(xiàn)在深層函數(shù)嵌套的錯誤情況時很有用處。
setjmp這個函數(shù)很有意思,雖然是一個函數(shù),可是卻可以返回兩個不同的值。當?shù)谝淮沃苯诱{(diào)用setjmp時,返回值為0。當從longjmp函數(shù)返回時,setjmp函數(shù)的返回值為longjmp的第二個參數(shù)的值。
那么在什么地方調(diào)用setjmp呢?我們希望當從longjmp函數(shù)返回時,程序從哪里接著開始運行,我們就在哪里調(diào)用setjmp。看個小實例,你就明白是怎么回事了。
#include<stdio.h>
#include<setjmp.h>
jmp_buf ebuf;
void f2(void);
int main(void)
{
int i;
printf("1");
i=setjmp(ebuf);
if(i==0) //第一次執(zhí)行到這里時,值為0,所以接下來執(zhí)行f2()
{
f2();
printf("This will not be printed.");
}
printf("%d",i); //由于從longjmp返回時,i=3,不執(zhí)行if,所以執(zhí)行該行
return 0;
}
void f2(void)
{
printf("2");
longjmp(ebuf,3); //longjmp函數(shù)返回,回到setjmp的位置,使得setjmp返回值為3
}
函數(shù)最后的執(zhí)行結果為123,嘻嘻。
longjmp注意:
1.不要假象寄存器類型的變量將總會保持不變。在調(diào)用longjmp之后,通過setjmp所返回的控制流中,例程中寄存器類型的變量將不會被恢復。
2.不要使用longjmp函數(shù)來實現(xiàn)把控制流,從一個中斷處理例程中傳出,除非被捕獲的異常是一個浮點數(shù)異常。在后一種情況下,如果程序通過調(diào)用 _fpreset函數(shù),來首先初始化浮點數(shù)包后,它是可以通過longjmp來實現(xiàn)從中斷處理例程中返回。
3.
在C++程序中,小心對setjmp和longjmp的使用,應為setjmp和longjmp并不能很好地支持C++中面向?qū)ο蟮恼Z義。因此在C++程
序中,使用C++提供的異常處理機制將會更加安全。把setjmp和longjmp組合起來,原來它這么厲害!