javascript - 在 Chrome 中操作图像时内存泄漏

标签 javascript jquery image google-chrome memory-leaks

我在 Chrome 中遇到了以下 2 个(巨大的!)内存泄漏:

  1. 使用新字节编辑现有图像的“src”时
  2. 使用 clone() 克隆图像时

请注意,在 Internet Explorer 中从来没有内存泄漏!

一些背景:我正在从事一个项目,其中外部摄像头提供实时图像馈送(假设每秒 100 帧)。

该项目的主要 3 个功能是:

  1. 播放直播
  2. 录制直播
  3. 显示录制的提要

欢迎您下载以下独立代码(只需将其保存为“leak.html”并执行),并亲自查看:

<!DOCTYPE html>
<html>
    <body>
        <canvas id="meCanvas" width="526" height="395"></canvas>

        <script src="http://code.jquery.com/jquery-2.0.3.min.js" type="text/javascript"> </script>
        <script>
            var meContext = document.getElementById("meCanvas").getContext("2d");

            // Bytes array representing a chair image
            var chairImgSrc = "";

            var image = new Image();
            image.onload = drawNewImage;

            var RECORD_LEN = 20;
            var recordedImages = new Array(RECORD_LEN);
            var count = 0;

            function drawNewImage() {
                meContext.clearRect(0, 0, meContext.canvas.width, meContext.canvas.height);
                meContext.drawImage(image, 0, 0, meContext.canvas.width, meContext.canvas.height);

                setTimeout(nextImage, 10); // Simulates 100 frames per second
            }

            function drawOldImage() {
                var curImage = count % RECORD_LEN; // Cyclic loop over the array
                meContext.clearRect(0, 0, meContext.canvas.width, meContext.canvas.height);
                meContext.drawImage(recordedImages[curImage], 0, 0, meContext.canvas.width, meContext.canvas.height);

                setTimeout(nextImage, 10); // Simulates 100 frames per second
            }

            function nextImage() {
                count++;
                if (count <= 1000) // Phase I (during first 10 seconds): use live camera feed
                {
                    // Generating a random image src (Just for this example!!, instead of using the real camera feed)
                    var newImgSrc = chairImgSrc.slice(0, -11) + ("00000" + count).slice(-6) + "/2Q==";
                    // (CHROME MEMORY LEAK #1: editing the 'src' of an existing image with new bytes creates a new memory that never gets released)
                    image.src = newImgSrc;

                    // Cloning the image, to keep a recorded array of the last N frames
                    var $tmpImage = $(image);
                    // (CHROME MEMORY LEAK #2: clone() creates a new memory that never gets released
                    var clonedImage = $tmpImage.clone()[0];
                    recordedImages[count % RECORD_LEN] = clonedImage;
                }
                else                // Phase II: use recorded feed
                {
                    drawOldImage();
                }
            }

            window.onload = nextImage;
        </script>
    </body>
</html>

此示例代码采用静态图像(椅子)并在每一帧随机修改它(只是为了模拟我的真实相机画面)。

对于前 1000 帧,它显示图像,并将最后 10 帧存储在循环数组中,从那时起它只显示最后记录的 10 帧(循环)。

(显然我的实际项目要复杂得多,我只是将其简化以说明问题)。

问题是 - 请建议一种替代方法(最好 - 基于提供的源代码)来执行完全相同的功能,而不会导致 Chrome 内存泄漏。

附注 1:

在 chromium 中,我发现了以下 2 个相关的错误,这些错误并没有真正修复(证据 - 我的代码仍然泄漏......):

  1. “通过 javascript 操作 img src 将产生大量内存泄漏”- https://code.google.com/p/chromium/issues/detail?id=36142
  2. “更改 img.src 时内存使用量无限增长”- https://code.google.com/p/chromium/issues/detail?id=114570

附注 2:

我完全了解 stackoverflow 中存在的类似问题,我做了很多尝试,但没有一个能帮助我解决我的问题:

  1. Rapidly updating image with Data URI causes caching, memory leak
  2. Canvas even Img eating RAM and CPU
  3. Refresh image with a new one at the same url
  4. Setting img.src to dataUrl Leaks Memory
  5. Memory leak when loading images with javascript's settimeout

我做的一些尝试,例如:

  • 为确保缓存不是原因,我在 chrome 的隐身模式下工作,因此缓存与此无关。
  • 我没有将字节数组设置为 src,而是尝试使用 blob URL(但仍然会发生类似的泄漏):
    • img.src = window.URL.createObjectURL(new Blob([bytes.buffer], {type : "image/jpeg"}));
  • 尝试将图像放在 iframe 中,并每隔 X 帧重新加载一次:这部分有帮助,但我几乎不可能使用这种“解决方法”。

* 更新 29/1 月 *

我替换了以下几行:

var $tmpImage = $(image);
var clonedImage = $tmpImage.clone()[0];

与:

var clonedImage = new Image();
clonedImage.src = newImgSrc;

和泄漏是一样的。

=> 所以我“只有”1 个需要解决方法的错误(在 2 个地方):编辑图像的 src 时泄漏。

最佳答案

我遇到了同样的问题。我发现的唯一解决方法是减少要使用的 new Image() 的数量(最好是一个):

function ImageLoader() {
  var img = new Image();
  var queue = [];
  var lock = false;
  var lastURL;
  var lastLoadOk;
  return { load: load };

  function load(url, callback, errorCallback) {
    if (lock) return queue.push(arguments);
    lock = true;
    if (lastURL === url) return lastLoadOk ? onload() : onerror();
    lastURL = url;
    img.onload = onload;
    img.onerror = onerror;
    img.src = url;

    function onload() {
      lastLoadOk = true;
      callback(img);
      oncomplete();
    }
    function onerror() {
      lastLoadOk = false;
      if (errorCallback) errorCallback(url);
      oncomplete();
    }
  }
  function oncomplete() {
    lock = false;
    if (queue.length) load.apply(null, queue.shift());
  }
}
var loader = new ImageLoader();
loader.load(url1, function(img) { // do something });
loader.load(url2, function(img) { // do something });

请注意,图像将按系列加载。如果要并行加载 2 个图像,则需要实例化 2 个 ImageLoader。

关于javascript - 在 Chrome 中操作图像时内存泄漏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21309760/

相关文章:

javascript - THREE.js - 无法读取未定义的属性 'lib' (THREE.ShaderUtils.lib ["normal"])

javascript - 什么是parent.result.location

javascript - 如何区分通过鼠标滚动和在 JavaScript 中以编程方式滚动?

javascript - 将更改文本 onclick 添加到 JavaScript 库

javascript - 如何防止签名 token 用户通过浏览器开发者工具访问 API 响应?

javascript - PaperJS随机点

javascript - SetTimeout 递归函数超出最大调用堆栈大小 (Javascript)

html - 使图像出现在悬停图像的其他位置

algorithm - 找到两张图像之间差异的边界框?

python - ImageTk.PhotoImage 无法正确显示图像