我可以撤消或删除atexit命令吗?

时间:2010-02-22 11:57:50

标签: c exit atexit longjmp

如果我将atexit( fn );放在退出堆栈上,它将在程序退出时执行:从main()exit()返回。

我可以从堆叠中删除它吗?

你问我为什么要这样做?

我正在使用atexitsetjmplongjmp尝试使用简单的try-catch机制。如果我能undo-atexit(fn);,那将是完美的 - 即使它只适用于最后注册的函数。

编辑:

遵循monoceres建议我自己的堆栈......

目前,堆栈仅适用于一个异常捕获器。

void (*_catchFn[10])()  = {0,0,0,0,0,0,0,0,0,0};

void _catch(){
  if ( _catchFn[0] != 0 ){
    (_catchFn[0])();
  }
}

void _addCatch( void (*fn)() ){
  _catchFn[0]=fn;
}

void _remCatch( void (*fn)() ){
  _catchFn[0]=0;
}

void test(){
  jmp_buf env;

  void catch(){                  // we get here after an exit with a registered catch
    longjmp(env,1);              // return to the line marked except...
                               //   that first will get the value 1
  }
  int first = setjmp( env);      // ** return here **
  fprintf( stderr , "test: After setjmp. first=%d\n" , first );
  if( first == 0 ){              // try this code
    _addCatch(catch);            // register the catch function to 'catch' the exit
    fprintf( stderr , "test: Before CHECK\n" );
    // CHECK something and something bad happens and it exits
    exit(1);                     // like this
    fprintf( stderr , "test: After CHECK - THIS SHOULD NEVER BE SEEN AFTER AN EXCEPTION.\n" );
  }else{
    fprintf( stderr , "test: After longjmp return. first=%d\n" , first );
  }
  _remCatch( catch);
  fprintf( stderr , "test: IT WORKED!\n");
  exit(1);  // exit again to see if we are safe
}

int main(){
  atexit( _catch );              // register my global exception stack
  test();
}

2 个答案:

答案 0 :(得分:14)

为什么不构建从单个atexit()函数调用的自己的堆栈?这样你就可以随心所欲地操纵堆栈。

答案 1 :(得分:7)

不,你不能这样做,但是你可以使用全局标志,这样如果设置了标志,你的退出处理程序就什么都不做。

或者你可以调用_Exit()(C99) - 它将执行正常的退出程序(关闭所有打开的描述符,发送所有需要的信号和父/子)但不会调用退出处理程序。