2

我正在创建一个供 10.6 及更高版本的 Cocoa 应用程序使用的框架。

该框架的目的是解析文本文件。

显然,可能会发生错误,例如找不到文件,权限问题等。

在框架内处理错误并通知宿主应用程序的正确方法是什么?

我的想法是:

  1. 什么都不做,让宿主应用程序捕获任何异常。

  2. 让宿主应用程序向框架注册它的第一个响应者,捕获任何异常,将它们转换为 NSError 并将它们传递给宿主应用程序的响应者链。

这些选项中的任何一个都有意义吗?还有其他选择吗?处理这个问题的正确方法是什么?

我已经阅读了错误和异常处理指南,但它们没有涵盖这种情况,只描述了应用程序本身的错误处理。

4

2 回答 2

2

我想说正确的方法是NSError在所有可能出错的方法中直接使用自己。我最近使用我创建的实用程序类完成了此操作,并且效果很好。然后,您允许应用程序决定如何处理错误(崩溃、日志、其他内容),并且框架无需担心。

以下是我用来创建错误对象的私有errno类方法,允许潜在的 POSIX 错误(等):

#pragma mark - Private Methods

- (NSError *)error:(NSString *)localizedDescription
              code:(EZipFileError)code
   underlyingError:(NSError *)underlyingError
{
    NSMutableDictionary *errorDetail = [NSMutableDictionary dictionary];
    [errorDetail setValue:localizedDescription forKey:NSLocalizedDescriptionKey];
    if (underlyingError != nil)
    {
        [errorDetail setValue:underlyingError forKey:NSUnderlyingErrorKey];
    }
    return [NSError errorWithDomain:@"MyErrorDomain"
                               code:(NSInteger)code
                           userInfo:errorDetail];
}

- (NSError *)error:(NSString *)localizedDescription
              code:(EZipFileError)code
{
    return [self error:localizedDescription
                  code:code
       underlyingError:nil];
}

- (NSError *)error:(NSString *)localizedDescription
              code:(EZipFileError)code
        posixError:(int)posixError
{
    NSMutableDictionary *underlyingErrorDetail = [NSMutableDictionary dictionary];
    [underlyingErrorDetail setValue:[NSString stringWithUTF8String:strerror(posixError)]
                             forKey:NSLocalizedDescriptionKey];
    NSError *underlyingError = [NSError errorWithDomain:NSPOSIXErrorDomain
                                                    code:posixError
                                                userInfo:underlyingErrorDetail];
    return [self error:localizedDescription
                  code:code
       underlyingError:underlyingError];
}

使用如下:

if (![self isOpen])
{
    if (error != NULL)
    {
        *error = [self error:@"File is not open"
                        code:ErrorNotOpen];
    }
    return nil;
}

这是一个使用该方法的底层 POSIX 错误版本的示例:

filefp = fopen([filename UTF8String], "rb");
if (filefp == NULL)
{
    if (error != NULL)
    {
        *error = [self error:@"Failed to open file"
                        code:ErrorOpenFileFailed
                  posixError:errno];
    }
    return NO;
}
于 2012-05-20T05:30:22.050 回答
1

异常应该只用于 Objective-C 中的终端错误。更具体地说,Cocoa 和 Cocoa Touch 不保证跨边界抛出的异常会从另一端出来,因此您不应该使用异常来处理错误。

向调用者报告错误的正确方法是通过 NSError 对象。您会注意到许多 Cocoa 和 Cocoa Touch 方法都包含一个 NSError* 参数来实现该目的。

于 2012-05-20T05:29:36.290 回答