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

时间:2014-01-23 13:37:41

标签: javascript jquery image google-chrome memory-leaks

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

  1. 编辑现有图像的'src'时,使用新字节
  2. 使用clone()克隆图像时
  3. 请注意,在Internet Explorer中没有任何内存泄漏!

    一些背景知识:我正在开展一个项目,在这个项目中,外部摄像头提供图像的实时反馈(假设每秒100帧)。

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

    1. 播放实时Feed
    2. 记录实时Feed
    3. 显示录制的Feed
    4. 欢迎您下载以下独立代码(只需将其另存为“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>
      

      此示例代码采用静态图像(椅子)并每帧随机修改(仅用于模拟我的真实相机Feed)。

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

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

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

      PS 1:

      在铬中我发现了以下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
      3. PS 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
        6. 我做了一些尝试,例如:

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

          *更新29 / Jan *

          我替换了以下几行:

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

          使用:

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

          和泄漏是一样的。

          =&GT;所以我要归结为“只有”一个需要解决方法的错误(在2个地方):编辑图像的src时出现泄漏。

1 个答案:

答案 0 :(得分:1)

我遇到了同样的问题。我找到的唯一解决方法是减少要使用的新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。