系统调用setjmp()和 longjmp()
有时候,当接收到一个信号时,希望能跳回程序中以前的一个位置执行。例如,在有的程序内,当用户按了中断键,则程序跳回到显示主菜单执行。我们可以用库系统调用setjmp()和 longjmp()来完成这项工作。setjmp()能保存程序中的当前位置(是通过保存堆栈环境实现的),longjmp()能把控制转回到被保存的位置。在某种意义上,longjmp()是远程跳转,而不是局部区域内的跳转。我们必须注意到,由于堆栈已经回到被保存位置这一点所以 longjmp()从来不返回。然而,与其对应的 setjmp()是要返回的。
setjmp()和 longjmp()在 setjmp.h中的定义分别如下:
- int setjmp(jmp_buf env);
- void longjmp(jmp_buf env, int val);
int setjmp(jmp_buf env); void longjmp(jmp_buf env, int val);
setjmp()只有一个参数 env,用来保存程序当前位置的堆栈环境。而 longjmp()有两个参数:
- 参数 env 是由setjmp()所保存的堆栈环境。
- 参数 val 设置 setjmp()的返回值。
longjmp()本身是没有返回的,但其执行后跳转到保存env 参数的 setjmp()调用,并由setjmp()调用返回,就好像程序刚刚执行完 setjmp()一样,此时 setjmp()的返回值就是 val。但是要注意的是,longjmp()调用不能使 setjmp()调用返回 0,如果 val为 0,则 setjmp()的返回为1。
下面的例子演示了 setjmp()和 longjmp()的使用:
- #include <stdio.h>
- #include <setjmp.h>
- #include <signal.h>
- jmp_buf position;
- main(){
- int goback();
- …
- …
- /* 保存当前的堆栈环境 */
- setjmp(position);
- signal(SIGINT,goback);
- domenu();
- …
- …
- }
- goback()
- {
- fprintf(stderr,” \nInterrupted\n” );
- /* 跳转回被保存的断点 */
- longjmp(position,1);
- }