4

我正在实现一个 Mac 应用程序,我想处理以下事件:

  • 未处理的异常
  • 程序崩溃(内存错误 dcc)

如果我检测到它们,我可以将详细信息发送给我,以使用我发现的其中一个崩溃处理程序来分析和修复错误。唉,我无法弄清楚如何拦截崩溃和异常。

  1. 第一个问题:我必须区分 Exceptions 和 Crashes 吗?还是检测异常就足够了?
  2. 如何捕获异常和/或崩溃,将它们重定向到我的处理程序?

PS我尝试在我的 MyApp 课程中关注

NSSetUncaughtExceptionHandler(&uncaughtExceptionHandler);
signal(SIGABRT, SignalHandler);
signal(SIGILL, SignalHandler);
signal(SIGSEGV, SignalHandler);
signal(SIGFPE, SignalHandler);
signal(SIGBUS, SignalHandler);
signal(SIGPIPE, SignalHandler);

但它不起作用。每次它崩溃时,它都会进入调试器而不对 SignalHandler 或 uncaughtExceptionHandler 进行分类

4

1 回答 1

3

我发现最好的方法是创建一个简单的异常处理委托类,因为这样IBAction可以捕获方法中的异常。

主.mm:

@interface ExceptionDelegate : NSObject
@end
static ExceptionDelegate *exceptionDelegate = nil; 


int main(int argc, char **argv)
{
    int retval = 1;

    @autoreleasepool
    {
        //
        // Set exception handler delegate
        //
        exceptionDelegate = [[ExceptionDelegate alloc] init];
        NSExceptionHandler *exceptionHandler = [NSExceptionHandler defaultExceptionHandler];
        exceptionHandler.exceptionHandlingMask = NSLogAndHandleEveryExceptionMask;
        exceptionHandler.delegate = exceptionDelegate;

        //
        // Set signal handler
        //
        int signals[] =
        {
            SIGQUIT, SIGILL, SIGTRAP, SIGABRT, SIGEMT, SIGFPE, SIGBUS, SIGSEGV,
            SIGSYS, SIGPIPE, SIGALRM, SIGXCPU, SIGXFSZ
        };
        const unsigned numSignals = sizeof(signals) / sizeof(signals[0]);
        struct sigaction sa;
        sa.sa_sigaction = signalHandler;
        sa.sa_flags = SA_SIGINFO;
        sigemptyset(&sa.sa_mask);
        for (unsigned i = 0; i < numSignals; i++)
            sigaction(signals[i], &sa, NULL);

        ....
    }

    ....

    return retval;    
}


static void signalHandler(int sig, siginfo_t *info, void *context)
{
    logerr(@"Caught signal %d", sig);
    exit(102);
}

@implementation ExceptionDelegate

- (BOOL)exceptionHandler:(NSExceptionHandler *)exceptionHandler
      shouldLogException:(NSException *)exception
                    mask:(unsigned int)mask
{
    logerr(@"An unhandled exception occurred: %@", [exception reason]);
   return YES;
}

- (BOOL)exceptionHandler:(NSExceptionHandler *)exceptionHandler
   shouldHandleException:(NSException *)exception
                    mask:(unsigned int)mask
{
    exit(101);

    // not reached
    return NO;
}

@end

您需要将 ExceptionHandling.framework 添加到您的项目中。

于 2012-11-29T08:13:55.147 回答