为什么在创建UIWebView时清除NSUserDefaults会导致EXC_CRASH?

时间:2012-03-13 06:10:35

标签: iphone objective-c ios uiwebview nsuserdefaults

在开始之前,我应该告诉您,此在iOS 5.1中发生。在最近的更新之前,这从未发生过,并且它仍然不会发生在任何其他版本上。也就是说,这是正在发生的事情。

当用户退出我的应用时,发生的一件事就是删除了所有NSUserDefaults。我不是手动删除可能添加到用户默认值的每个键,而是使用this SO question中建议的方法完全删除所有NSUserDefaults

NSString *appDomain = [[NSBundle mainBundle] bundleIdentifier];
[[NSUserDefaults standardUserDefaults] removePersistentDomainForName:appDomain];

似乎发生的事情是,每次我在删除UIWebView后尝试创建NSUserDefaults时,我都会得到EXC_CRASH (SIGABRT)。当我拨打[[UIWebView alloc] initWithFrame:frame]时发生了崩溃。奇怪吧?完全退出并重新打开应用程序可以再次创建UIWebView

所以,我设法找出删除默认值会导致UIWebView问题,但可以肯定的是,我为-[NSUserDefaults setObject:forKey:]添加了一个符号断点。

-[NSUserDefaults setObject:forKey:] breakpoint

创建UIWebView确实会触发断点。

查看崩溃日志会给我一个例外原因:

-[__NSCFDictionary setObject:forKey:]: attempt to insert nil value (key: WebKitLocalStorageDatabasePathPreferenceKey)

这是堆栈跟踪的开始:

0 CoreFoundation 0x3340688f __exceptionPreprocess + 163
1 libobjc.A.dylib 0x37bd4259 objc_exception_throw + 33
2 CoreFoundation 0x33406789 +[NSException raise:format:] + 1
3 CoreFoundation 0x334067ab +[NSException raise:format:] + 35
4 CoreFoundation 0x3337368b -[__NSCFDictionary setObject:forKey:] + 235
5 WebKit 0x3541e043 -[WebPreferences _setStringValue:forKey:] + 151
6 UIKit 0x32841f8f -[UIWebView _webViewCommonInit:] + 1547
7 UIKit 0x328418d7 -[UIWebView initWithFrame:] + 75
8 MyApp 0x0007576f + 0
9 UIKit 0x326d4dbf -[UIViewController view] + 51
10 UIKit 0x327347e5 -[UITabBarController transitionFromViewController:toViewController:transition:shouldSetSelected:] + 93
11 UIKit 0x32734783 -[UITabBarController transitionFromViewController:toViewController:] + 31
12 UIKit 0x327340bd -[UITabBarController _setSelectedViewController:] + 301
13 UIKit 0x327bd5d9 -[UITabBarController _tabBarItemClicked:] + 345

我现在正在做的,有效的是,只需跟踪我设置的NSUserDefaults键,并在需要时手动删除它们。但总有一种风险,我可能会忘记一个敏感键,所以简单地清除所有NSUserDefaults让我觉得更明智。所以,我想知道为什么我不能这样做。这是一个错误还是我做错了什么?

如果您想了解更多信息,请告诉我们!感谢。

编辑:删除所有[[NSUserDefaults standardUserDefaults] synchronize]后执行NSUserDefaults无效。

4 个答案:

答案 0 :(得分:25)

我也看到了这个问题,我认为你必须首先创建一个UIWebView,然后才能清除用户默认值。带有以下内容的干净项目将导致iOS5.1崩溃,这在iOS5.0及更早版本中运行良好:

UIWebView *webView = [[UIWebView alloc] init];
[webView release];

[[NSUserDefaults standardUserDefaults] setPersistentDomain:[NSDictionary dictionary] forName:[[NSBundle mainBundle] bundleIdentifier]];

UIWebView *anotherWebView = [[UIWebView alloc] init];
[anotherWebView release];

我可以通过这样做来解决崩溃,这可以避免记住所有设置键:

id workaround51Crash = [[NSUserDefaults standardUserDefaults] objectForKey:@"WebKitLocalStorageDatabasePathPreferenceKey"];
NSDictionary *emptySettings = (workaround51Crash != nil)
                ? [NSDictionary dictionaryWithObject:workaround51Crash forKey:@"WebKitLocalStorageDatabasePathPreferenceKey"]
                : [NSDictionary dictionary];
[[NSUserDefaults standardUserDefaults] setPersistentDomain:emptySettings forName:[[NSBundle mainBundle] bundleIdentifier]];

任何人都会看到这样做的任何问题吗?

答案 1 :(得分:3)

听起来你正在删除某种私人偏好,这可能是一个新的bug,无论是NSUserDefaults(你都不能删除它)或UIWebView(它应该处理缺少的条目)。

您是否尝试过其他答案中的方法(设置空字典?)。这会产生相同的结果吗?

如果获得NSUserDefaults的字典表示,获取所有键并迭代这些键,删除对象,怎么样? (如果您需要代码示例,请告诉我)。

答案 2 :(得分:0)

它对我有用

NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];

NSDictionary *userDefaultsDictionary = [userDefaults dictionaryRepresentation];
NSString *strWebDatabaseDirectory = [userDefaultsDictionary objectForKey:@"WebDatabaseDirectory"];
NSString *strWebKitLocalStorageDatabasePathPreferenceKey = [userDefaultsDictionary objectForKey:@"WebKitLocalStorageDatabasePathPreferenceKey"];

[userDefaults removePersistentDomainForName:[[NSBundle mainBundle] bundleIdentifier]];

if (strWebDatabaseDirectory) {
    [userDefaults setObject:strWebDatabaseDirectory forKey:@"WebDatabaseDirectory"];}
if (strWebKitLocalStorageDatabasePathPreferenceKey) {
    [userDefaults setObject:strWebKitLocalStorageDatabasePathPreferenceKey forKey:@"WebKitLocalStorageDatabasePathPreferenceKey"];}

[userDefaults synchronize];

答案 3 :(得分:0)

更简单的是使用以下代码:

[self saveValue:@"" forKey:@"WebKitLocalStorageDatabasePathPreferenceKey"];
[[NSUserDefaults standardUserDefaults] synchronize];

这更容易。