我正在努力捕捉我的应用程序中的错误,我正在寻找使用NSError。对于如何使用它,以及如何填充它,我有点困惑。
有人能提供一个例子,我如何填充然后使用NSError?
我正在努力捕捉我的应用程序中的错误,我正在寻找使用NSError。对于如何使用它,以及如何填充它,我有点困惑。
有人能提供一个例子,我如何填充然后使用NSError?
当前回答
我将尝试总结Alex的伟大回答和jlmendezbonini的观点,添加一个修改,使所有内容都与ARC兼容(到目前为止,它不是因为ARC会抱怨,因为你应该返回id,这意味着“任何对象”,但BOOL不是对象类型)。
- (BOOL) endWorldHunger:(id)largeAmountsOfMonies error:(NSError**)error {
// begin feeding the world's children...
// it's all going well until....
if (ohNoImOutOfMonies) {
// sad, we can't solve world hunger, but we can let people know what went wrong!
// init dictionary to be used to populate error object
NSMutableDictionary* details = [NSMutableDictionary dictionary];
[details setValue:@"ran out of money" forKey:NSLocalizedDescriptionKey];
// populate the error object with the details
if (error != NULL) {
// populate the error object with the details
*error = [NSError errorWithDomain:@"world" code:200 userInfo:details];
}
// we couldn't feed the world's children...return nil..sniffle...sniffle
return NO;
}
// wohoo! We fed the world's children. The world is now in lots of debt. But who cares?
return YES;
}
现在,我们不再检查方法调用的返回值,而是检查error是否仍然为nil。如果不是,我们就有麻烦了。
// initialize NSError object
NSError* error = nil;
// try to feed the world
BOOL success = [self endWorldHunger:smallAmountsOfMonies error:&error];
if (!success) {
// inspect error
NSLog(@"%@", [error localizedDescription]);
}
// otherwise the world has been fed. Wow, your code must rock.
其他回答
我将尝试总结Alex的伟大回答和jlmendezbonini的观点,添加一个修改,使所有内容都与ARC兼容(到目前为止,它不是因为ARC会抱怨,因为你应该返回id,这意味着“任何对象”,但BOOL不是对象类型)。
- (BOOL) endWorldHunger:(id)largeAmountsOfMonies error:(NSError**)error {
// begin feeding the world's children...
// it's all going well until....
if (ohNoImOutOfMonies) {
// sad, we can't solve world hunger, but we can let people know what went wrong!
// init dictionary to be used to populate error object
NSMutableDictionary* details = [NSMutableDictionary dictionary];
[details setValue:@"ran out of money" forKey:NSLocalizedDescriptionKey];
// populate the error object with the details
if (error != NULL) {
// populate the error object with the details
*error = [NSError errorWithDomain:@"world" code:200 userInfo:details];
}
// we couldn't feed the world's children...return nil..sniffle...sniffle
return NO;
}
// wohoo! We fed the world's children. The world is now in lots of debt. But who cares?
return YES;
}
现在,我们不再检查方法调用的返回值,而是检查error是否仍然为nil。如果不是,我们就有麻烦了。
// initialize NSError object
NSError* error = nil;
// try to feed the world
BOOL success = [self endWorldHunger:smallAmountsOfMonies error:&error];
if (!success) {
// inspect error
NSLog(@"%@", [error localizedDescription]);
}
// otherwise the world has been fed. Wow, your code must rock.
这有点超出了问题范围,但如果你没有NSError选项,你可以总是显示低级别错误:
NSLog(@"Error = %@ ",[NSString stringWithUTF8String:strerror(errno)]);
objective - c
NSError *err = [NSError errorWithDomain:@"some_domain"
code:100
userInfo:@{
NSLocalizedDescriptionKey:@"Something went wrong"
}];
斯威夫特3
let error = NSError(domain: "some_domain",
code: 100,
userInfo: [NSLocalizedDescriptionKey: "Something went wrong"])
基于我最近的实现,我想添加更多的建议。我看过一些来自苹果的代码,我认为我的代码的行为方式大致相同。
上面的文章已经解释了如何创建NSError对象并返回它们,所以我就不麻烦这部分了。我只是试图建议一个好方法来集成错误(代码,消息)在你自己的应用程序。
我建议创建一个标题,这将是你的域(即应用程序,库等)的所有错误的概述。我的当前标题是这样的:
FSError.h
FOUNDATION_EXPORT NSString *const FSMyAppErrorDomain;
enum {
FSUserNotLoggedInError = 1000,
FSUserLogoutFailedError,
FSProfileParsingFailedError,
FSProfileBadLoginError,
FSFNIDParsingFailedError,
};
FSError.m
#import "FSError.h"
NSString *const FSMyAppErrorDomain = @"com.felis.myapp";
现在,当使用上述错误值时,苹果将为你的应用程序创建一些基本的标准错误消息。错误可能如下所示:
+ (FSProfileInfo *)profileInfoWithData:(NSData *)data error:(NSError **)error
{
FSProfileInfo *profileInfo = [[FSProfileInfo alloc] init];
if (profileInfo)
{
/* ... lots of parsing code here ... */
if (profileInfo.username == nil)
{
*error = [NSError errorWithDomain:FSMyAppErrorDomain code:FSProfileParsingFailedError userInfo:nil];
return nil;
}
}
return profileInfo;
}
苹果为上述代码生成的标准错误消息(error. localizeddescription)如下所示:
域= com.felis错误。myapp Code=1002 "操作无法完成。(com.felis。Myapp错误1002)”
上面的内容对开发人员已经非常有帮助,因为消息显示了发生错误的域和相应的错误代码。最终用户将不知道错误代码1002意味着什么,所以现在我们需要为每个代码实现一些良好的消息。
对于错误消息,我们必须牢记本地化(即使我们没有立即实现本地化消息)。在我目前的项目中,我使用了以下方法:
1)创建一个包含错误的字符串文件。字符串文件很容易本地化。该文件看起来如下所示:
FSError.strings
"1000" = "User not logged in.";
"1001" = "Logout failed.";
"1002" = "Parser failed.";
"1003" = "Incorrect username or password.";
"1004" = "Failed to parse FNID."
2)添加宏,将整数码转换为本地化的错误消息。我在Constants+ macro .h文件中使用了2个宏。为了方便起见,我总是将这个文件包含在前缀头(MyApp-Prefix.pch)中。
常量+ Macros.h
// error handling ...
#define FS_ERROR_KEY(code) [NSString stringWithFormat:@"%d", code]
#define FS_ERROR_LOCALIZED_DESCRIPTION(code) NSLocalizedStringFromTable(FS_ERROR_KEY(code), @"FSError", nil)
3)现在很容易根据错误代码显示用户友好的错误消息。一个例子:
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Error"
message:FS_ERROR_LOCALIZED_DESCRIPTION(error.code)
delegate:nil
cancelButtonTitle:@"OK"
otherButtonTitles:nil];
[alert show];
答得好,亚历克斯。一个潜在的问题是NULL解引用。Apple关于创建和返回NSError对象的参考
...
[details setValue:@"ran out of money" forKey:NSLocalizedDescriptionKey];
if (error != NULL) {
// populate the error object with the details
*error = [NSError errorWithDomain:@"world" code:200 userInfo:details];
}
// we couldn't feed the world's children...return nil..sniffle...sniffle
return nil;
...