javascript - 动态 iframe src 更改导致内存泄漏
问题描述
我正在测试模板内部src
属性的动态变化。在测试中,我每秒更改一次 src。在几分钟之内,我的可用内存被 88% 使用,最终计算机最终死机。我在具有 32 GB 内存和 Firefox 浏览器的 Windows 10 机器上执行了此测试。<iframe>
Vue
但是,我首先注意到在使用 chromium-browser 的 Raspberry Pi 4 (4GB) 上使用类似方法的内存问题。我不断地切换多个 vue 组件(如幻灯片)。其中一些也有iframes
。就像上面的测试一样,内存泄漏(不是那么快,但在几天内),并且铬浏览器选项卡崩溃了,显示了他死去的吉姆笑脸。
这是来自测试的代码:
<template>
<div id="app">
<iframe :src="src" />
{{ count }}
{{ src }}
</div>
</template>
<script>
export default {
data() {
return {
src : "",
source : [
"https://example.com/embeddedcontent/1",
"https://example.com/embeddedcontent/2",
"https://example.com/embeddedcontent/3",
"https://example.com/embeddedcontent/4"
],
count : 0,
interval : null
}
},
mounted() {
const interval = setInterval(() => {
this.src = this.source[ this.count % this.source.length ];
this.count = this.count + 1
}, 2000);
this.interval = interval;
},
beforeDestroy() {
clearInterval(this.interval);
}
}
</script>
事实
- 注意到不同系统和浏览器上的相同内存泄漏 (
Win10/Firefox
,Raspbian/chromium-browser
) - 在 Vue 的开发 (HMR) 和生产构建过程中注意到相同的内存泄漏。不过,单独的 Vue 应用程序。
- iframe 的内容对我来说是未知的,可以是任何内容,因为该
src
属性可以设置为任何 URL。
有什么想法可以防止这种情况吗?
到目前为止,我只想每隔几分钟刷新一次浏览器选项卡。这将是一种 hacky 解决方案,因为它没有解决问题的根源。iframes
当我想在我的 Vue 应用程序中显示其他网站内容时,是否有任何替代方法可以使用?有没有办法清理整个iframe
而不留下任何东西?
更新 2021/02/25
我现在也尝试了 Edge Browser,结果相同(增加内存超时)。我还尝试了 Firefox Private 模式,但影响很小(增加内存但速度有点慢)。比较了 vue 示例的生产和开发版本,没有区别。我还尝试了一个带有以下代码的香草电子应用程序(无 vue):
let interval = null;
let sources = [
"https://hurtigruten.panomax.com/ms-roald-amundsen",
"https://hurtigruten.panomax.com/ms-fridtjof-nansen",
"https://rosenalp.panomax.com/",
"https://alpbach.panomax.com/galtenberg",
];
let index = 0;
let count = 0;
function startTheInterval() {
interval = setInterval(() => {
index = count % sources.length;
count = count + 1;
document.getElementById('monitor').src = sources[index];
},2000);
}
function clearTheInterval() {
clearInterval(interval);
}
startTheInterval();
同样的结果,快速增加内存。
解决方案
首先,关于浏览器如何iframe
工作的一些理论 -分离窗口内存泄漏
当您尝试调试内存泄漏时,请始终使用“隐身模式” - 当浏览器不使用任何扩展时的模式。扩展可以保持对加载到 a 中的 HTML 的引用
iframe
并保持该数据处于活动状态。当我在 Chrome 中分析您的示例时,只是切换到隐身模式会导致内存分配和保留的巨大减少......不要使用 Vue CLI/Webpack 开发模式来调试内存泄漏。Webpack 的热模块重载 (HMR) 会影响内存中保存的内容。始终为此使用生产版本...
请注意,JS 运行时(在本例中为 V8)收集有关正在执行的代码的一些元数据并将其保存在 JS heap中。看起来像内存泄漏的可能只是 JS 虚拟机元数据......
更新:我已经在我的开发机器(Ryzen 5 3600、32G RAM、Win 10 x64)上使用您的示例(间隔 1 秒而不是 2 秒)运行了一些测试 - 生产构建通过serve-handler
加载到 Firefox 开发人员版 v86.0b9 ( 64 位)私有窗口(所以没有任何类型的扩展)
首先打开开发工具并打开内存分析(打开
Record call stacks
选项)。开始时录制一张快照,大约 40 分钟后再录制一张。比较这些快照并没有显示任何 JS 堆分配增加。浏览器的私有字节(使用 Sysinternals Process Explorer)显示增加了大约 60MB,但是在我关闭开发工具的那一刻,内存被清除了,所以可以肯定地说它是开发工具使用的内存相同的示例,现在无需打开开发工具 - 运行 40 分钟。再次没有任何内存增加......
所以我的结论是没有与iframe
Vue 本身相关的内存泄漏。如果你真的看到像“在几分钟内我的可用内存被 88% 使用”这样戏剧性的东西,你应该检查你自己的系统,尤其是浏览器中安装的扩展......
推荐阅读
- android - Firebase 中有没有办法自动将 id 列表交换到对象中
- mysql - 多自动增量
- excel - Excel 3D 地图 - 重新排序图层
- java - Hadoop 与 openjdk:start-dfs.sh 错误(SSH?)
- blender - JAVA3D 加载从搅拌机导出的 .obj 文件
- vba - 基于列表删除VBA中的行
- jquery - 无法使用 .html() 功能限制获取父 div 的完整内容
- angular5 - Angular 5中的RequestOptions不推荐使用符号错误
- properties - 更改属性值时,mapbox会动态更改圆圈颜色
- angular - 定义 json 对象响应的接口