Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/jquery/88.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Javascript 在Chrome中操作图像时内存泄漏_Javascript_Jquery_Image_Google Chrome_Memory Leaks - Fatal编程技术网

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

Javascript 在Chrome中操作图像时内存泄漏,javascript,jquery,image,google-chrome,memory-leaks,Javascript,Jquery,Image,Google Chrome,Memory Leaks,我在Chrome中遇到了以下2个(巨大!)内存泄漏: 编辑现有图像的“src”时,使用新字节 使用clone()克隆图像时 请注意,在InternetExplorer中没有内存泄漏 一些背景:我正在做一个项目,在这个项目中,一个外部摄像头提供了一个实时的图像馈送(比如说每秒100帧) 该项目的主要3项功能是: 播放实况转播 录制实时提要 显示录制的提要 欢迎您下载以下独立代码(只需将其保存为“leak.html”并执行),然后亲自查看: <!DOCTYPE html> <htm

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

  • 编辑现有图像的“src”时,使用新字节
  • 使用clone()克隆图像时
  • 请注意,在InternetExplorer中没有内存泄漏

    一些背景:我正在做一个项目,在这个项目中,一个外部摄像头提供了一个实时的图像馈送(比如说每秒100帧)

    该项目的主要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>
    
    与:

    泄漏是一样的


    =>所以我只发现了一个需要解决的bug(在两个地方):编辑图像的src时泄漏。

    我也遇到了同样的问题。我发现的唯一解决办法是减少要使用的新映像()的数量(理想情况下是一个):


    请注意,图像将以序列形式加载。如果你想并行加载2个图像,你需要实例化2个ImageLoader。

    问题是?..@HAL9000-你说得对,现在就编辑我的帖子来反映它。为了防止我找不到解决方法,我在Chrome的bug跟踪系统中并行发布了一个问题:几乎如果你只使用一个图像对象。你不能做任何其他事情,因为这是一个chrome错误,但使用此解决方案,它使用的内存确实更少(100Mb而不是3gb)。此解决方案可能在chrome中实现此目的,但它不能解决Safari中的内存泄漏问题。在当前版本中,即使只使用一个图像对象,内存也会不断增加。在最新版本的electron(9.0.5)中,它使用铬83,这似乎也不能解决内存泄漏问题(内存不断增加,从300mb增加到>3.5gb!)。下一步我将尝试:
    var $tmpImage = $(image);
    var clonedImage = $tmpImage.clone()[0];
    
    var clonedImage = new Image();
    clonedImage.src = newImgSrc;
    
    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 });