setjmp 与 longjmp

setjmp和longjmp是C语言独有的,只有将它们结合起来使用,才能达到程序控制流有效转移的目的,按照程序员的预先设计的意图,去实现对程序中可能出现的异常进行集中处理。程序员

 

setjmp和longjmp的函数原型在setjmp.h中函数

函数原型:设计

int setjmp(jmp_buf envbuf);

setjmp函数用缓冲区envbuf保存系统堆栈的内容,以便后续的longjmp函数使用。setjmp函数初次启用时返回0值。code

 

void longjmp(jmp_buf envbuf, int val);

longjmp函数中的参数envbuf是由setjmp函数所保存的堆栈环境,参数val设置setjmp函数的返回值。longjmp函数自己是没有返回值的,它执行后跳转到保存envbuf参数的setjmp函数调用,并由setjmp函数调用返回,此时setjmp函数的返回值就是val。原型

jmp_buf 记录了栈、指令位置等信息:it

1,若是有多个setjmp 使用同一个jmp_buf, 那么longjmp会跳转到最近的setjmp位置: class

#include "apue.h"
#include <setjmp.h>

static void	f1(int, int, int, int);
static void	f2(void);

static jmp_buf	jmpbuffer;
static int	globval;

int
main(void)
{
	int				autoval;
	register int	regival;
	volatile int	volaval;
	static int		statval;

	globval = 1; autoval = 2; regival = 3; volaval = 4; statval = 5;

	if (setjmp(jmpbuffer) != 0) {
		printf("in main,after longjmp:\n");
		printf("globval = %d, autoval = %d, regival = %d,"
		    " volaval = %d, statval = %d\n",
		    globval, autoval, regival, volaval, statval);
		exit(0);
	}

	/*
	 * Change variables after setjmp, but before longjmp.
	 */
	globval = 95; autoval = 96; regival = 97; volaval = 98;
	statval = 99;

	f1(autoval, regival, volaval, statval);	/* never returns */
	exit(0);
}

static void
f1(int i, int j, int k, int l)
{
	printf("in f1():\n");
	printf("globval = %d, autoval = %d, regival = %d,"
	    " volaval = %d, statval = %d\n", globval, i, j, k, l);

    if(setjmp(jmpbuffer) != 0){
        printf("in f1, after longjmp!");
        //longjmp(jmpbuffer,2);
        exit(0);
    }
    
	f2();
}

static void
f2(void)
{
    printf("in f2");
    longjmp(jmpbuffer, 1);
}

 

2, 下面的使用方式会造成死循环:循环

if( setjmp(jmpbuffer) != 0 ){
    printf("in f1, after longjmp!");
    longjmp(jmpbuffer,2);
    exit(0);
}

 

3, 若是想实现多个不一样的跳转,可使用多个jmp_buf :程序

#include "apue.h"
#include <setjmp.h>

static void	f1(int, int, int, int);
static void	f2(void);

static jmp_buf	jmpbuffer;
static jmp_buf	jmpbuffer2;
static int		globval;

int
main(void)
{
	int				autoval;
	register int	regival;
	volatile int	volaval;
	static int		statval;

	globval = 1; autoval = 2; regival = 3; volaval = 4; statval = 5;

	if (setjmp(jmpbuffer2) == 2) {
		printf("\n in main,after longjmp:\n");
		printf("globval = %d, autoval = %d, regival = %d,"
		    " volaval = %d, statval = %d\n",
		    globval, autoval, regival, volaval, statval);
		exit(0);
	}

	/*
	 * Change variables after setjmp, but before longjmp.
	 */
	globval = 95; autoval = 96; regival = 97; volaval = 98;
	statval = 99;

	f1(autoval, regival, volaval, statval);	/* never returns */
	exit(0);
}

static void
f1(int i, int j, int k, int l)
{
	printf("in f1():\n");
	printf("globval = %d, autoval = %d, regival = %d,"
	    " volaval = %d, statval = %d\n", globval, i, j, k, l);

    if(setjmp(jmpbuffer) != 0){
        printf("in f1, after longjmp!\n");
        longjmp(jmpbuffer2,2);
        exit(0);
    }
    
	f2();
}

static void
f2(void)
{
    printf("in f2\n");
	longjmp(jmpbuffer, 1);
}

 

相应的输出:异常

in f1():
globval = 95, autoval = 96, regival = 97, volaval = 98, statval = 99
in f2
in f1, after longjmp!

 in main,after longjmp:
globval = 95, autoval = 96, regival = 97, volaval = 98, statval = 99
相关文章
相关标签/搜索