10

我正在尝试修复我UIWebView造成的一堆泄漏,但找不到它们的来源,也找不到解决方法。我所做的是通过网络请求从 Web 获取一些内容,然后组装我的 HTML 并动态加载它:

NSString* body = <some HTML>;
NSString* html = [NSString stringWithFormat:kHTMLTemplate, [self scripts], [self styles], body];
[_webView loadHTMLString:html
               baseURL:[NSURL fileURLWithPath:[[NSBundle mainBundle] bundlePath]]];

每次有新内容可用时,我都会loadHTMLString再次执行以刷新 Web 视图。我重复使用相同的 Web 视图、相同的控制器、相同的一切。

Instruments 显示了一种非常奇怪的模式,其中所有泄漏的对象都是各种大小的通用块,并且没有一个附加任何信息:没有负责的库,没有负责的框架等。每次loadHTMLString执行时,都会添加新的泄漏。

SO中似乎有几个关于UIWebView泄漏内存的线程。我已经尝试了我找到的所有建议(例如,将 设置NSURLCache为零或重置它;我尝试释放现有的 UIWebView 并在每次有新数据时分配一个新的,等等)但没有任何帮助。

到目前为止,我的调查得出了一个明确的结果:似乎只有当我加载到视图中的 HTML 包含一些 Javascript 时才会出现泄漏。如果你注意到html上面的字符串,它是由几个组件组成的;一个是[self scripts]简单返回的函数:

return @"<script type='text/javascript' src='jquery-1.4.4.min.js'></script>"
        "<script type='text/javascript' src='jmy.js'></script>";

如果我删除它,就没有泄漏。但是,一旦我<script>向我的 HTML 添加标签,就会出现泄漏。如果我简单地包含 jquery 文件(或任何其他 js 文件,至于这个),它们甚至会出现:

return @"<script type='text/javascript' src='jquery-1.4.4.min.js'></script>";

所以,问题是:有人知道这里发生了什么吗?显然,在我的 HTML 中包含一个 Javascript 文件会UIWebView导致内存泄漏。

UIWebView当我重用同一个对象或每次我有内容时实例化一个新对象时都会出现泄漏这一事实,这使我认为 javascript 文件的处理方式一定存在loadHTMLString导致泄漏的原因。

有谁知道如何解决这个问题?

在此处输入图像描述

4

1 回答 1

11

我终于找到了一些关于正在发生的事情的线索,最重要的是我想分享一个解决方法。

我可以确认一些 javascript 文件的简单包含导致重新加载 Web 视图时出现内存泄漏。我什至尝试用 HTML 内容构建一个文件,然后将其加载到UIWebViewthroughloadRequest中,然后重新加载通过reload; 泄漏总是存在的。我会为此发布一个雷达。

拯救我的是innerHTML用来更新网络视图的内容。我没有依赖reloador loadHTMLString,而是用一个空的主体初始化了我的 web 视图(我的意思是,该head部分在那里,包括所有必需的 JS/CSS 文件)然后更新了它的设置document.body.innerHTML

body = [body stringByReplacingOccurrencesOfString:@"\"" withString:@"\\\""];
[webView stringByEvaluatingJavaScriptFromString:[NSString stringWithFormat:@"setBody(\"%@\");", body]];

setBody 定义如下:

var setBody = function(body) {
    document.body.innerHTML = body;
}

我获得了两个好处:Web 视图更新变得非常快(这是不更新 DOM 的效果,另一方面,这在整体上并不完全可取),并且在 Instruments 下运行应用程序时没有内存泄漏。缺点是我必须微调应用程序运行良好的几个条件。具体来说:

  1. 加载 web 视图(即使是空的正文页面)需要很多时间,因此您必须将其内容的第一次更新同步到 DOM 准备好时;

  2. webViewDidFinishLoading似乎不可靠:它在document.readyState成为之前执行complete

  3. document.documentElement.height,检索页面高度的官方方法似乎也不可靠:解决方法是获取body部件的“计算样式”并读取其height值。

希望这可以帮助其他发现他的网络视图正在泄漏内存的人。

于 2012-08-07T15:39:18.380 回答