为啥在创建 UIWebView 后清除 NSUserDefaults 会导致 EXC_CRASH?

Posted

技术标签:

【中文标题】为啥在创建 UIWebView 后清除 NSUserDefaults 会导致 EXC_CRASH?【英文标题】:Why does clearing NSUserDefaults cause EXC_CRASH later when creating a UIWebView?为什么在创建 UIWebView 后清除 NSUserDefaults 会导致 EXC_CRASH? 【发布时间】:2012-03-13 06:10:34 【问题描述】:

在我开始之前,我应该告诉你这发生在 ios 5.1 中。在最近的更新之前,这从未发生过,并且在任何其他版本上仍然没有发生。也就是说,这就是发生的事情。

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

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

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

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

创建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 后执行 [[NSUserDefaults standardUserDefaults] synchronize] 无济于事。

【问题讨论】:

显示更多您在同一个控制器中使用的代码。 UIWebView 是否依赖于存储在首选项中的任何值? EXC_CRASH 发生独立于哪个控制器正在加载 UIWebView。我将所有 UIWebView 初始化都包装在 @try @catch 块中,每个都给了我同样的例外。我想不出 UIWebView 可能依赖的任何值,尤其是在初始化时。如果您仍然想要包含 UIWebView 的控制器中的一些代码,请告诉我。 是的,发布所有相关代码。 伙计,我不知道你是如何在 UIWebView 和 NSUserDefaults 之间建立联系的,但你只是为我节省了令人沮丧的 3 个小时 【参考方案1】:

我也看到了这个问题,我认为您必须先创建一个 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]];

有人发现这样做有什么问题吗?

【讨论】:

啊,是的,它重现了这个错误。谢谢你的提示!此外,这似乎是避免崩溃的合理方法;我一直在使用它,并没有发现它有任何问题。干杯:) 只是这样做了,它似乎工作。我相信我们在管理用户注销时都会遇到这个问题。 这是一个很好的修复,但看起来太具体了,并且可能无法防范系统可能依赖的其他未知键。花了一些时间,但我收集了我正在使用的每一个密钥,并在用户注销时手动删除了每一个。 用户默认值中不应有任何系统关键键。这是针对明显是 Apple 错误的解决方法。但是,如果您预计将来会有更多此类错误,您当然可以跟踪自己的密钥。 试一试,但不起作用...我仍然崩溃...有什么想法吗?【参考方案2】:

听起来您正在删除某种私人偏好,这可能是一个新错误,无论是使用 NSUserDefaults(您不应该能够删除它)还是 UIWebView(它应该可以处理丢失的条目)。

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

如果你得到 NSUserDefaults 的字典表示,得到所有的键,然后遍历它们,删除对象呢? (如果您需要代码示例,请告诉我)。

【讨论】:

这是有道理的。我尝试了另一种设置空白字典的方法,但无济于事。我刚刚尝试了您关于遍历所有键的建议,这会产生相同的结果。查看字典,我看到一堆我没有设置的键,主要是WebKit* 格式。 WebKitLocalStorageDatabasePathPreferenceKey 就是其中之一。 这听起来像是一个错误,那么。如果您在 iOS 5.0 或更早版本上运行,您看到的键是否不同?提交错误时将需要此类信息。 很抱歉延迟回复您。我创建了一个干净的应用程序,在 NSUserDefaults 中设置一个键,然后删除所有 NSUserDefaults,然后尝试创建 UIWebView。我还无法重现那里的错误,所以我只能假设我的应用程序中出现了其他问题。我将提交错误报告并继续尝试重现该错误。我会在这里发布任何更新。谢谢!【参考方案3】:

对我有用

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];

【讨论】:

【参考方案4】:

使用下面的代码会更简单:

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

这样更容易。

【讨论】:

以上是关于为啥在创建 UIWebView 后清除 NSUserDefaults 会导致 EXC_CRASH?的主要内容,如果未能解决你的问题,请参考以下文章

清除 UIWebview 缓存

iOS 清除 UIWebView 的回溯历史

如何清除/禁用 UIWebView 的浏览历史记录

如何清除 UIWebView/WKWebView 的内容?

如何在 UISplitView 的详细视图中清除 UIWebView 的内容

为啥清除对象的内容不会释放内存?