Przecieka pamięć podczas manipulowania obrazami w Chrome

Napotkałam następujące 2 (ogromne!) Wycieki pamięci w Chrome:

Podczas edycji 'src' istniejącego obrazu, z nowymi bajtamiGdy używasz klonu () do sklonowania obrazu

Zauważ, że w Internet Explorerze nie ma wycieku pamięci, co kiedykolwiek!

Trochę tła: pracuję nad projektem, w którym zewnętrzna kamera zapewnia podgląd obrazu na żywo (powiedzmy 100 klatek na sekundę).

Główne 3 funkcje projektu to:

grać na żywonagrywać transmisję na żywopokaż nagrany kanał

Możesz pobrać następujący samodzielny kod (po prostu zapisz go jako „leak.html” i uruchom go) i przekonaj się sam:

<!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>

Ten przykładowy kod pobiera statyczny obraz (krzesła) i modyfikuje go losowo w każdej klatce (tylko w celu symulacji mojegoreal przekaz kamery).

Przez pierwsze 1000 klatek pokazuje obraz i przechowuje ostatnie 10 klatek w cyklicznej tablicy, a od tego momentu pokazuje tylko 10 ostatnio nagranych klatek (w pętli).

(Oczywiście mój prawdziwy projekt jest znacznie bardziej skomplikowany, po prostu uprościłem go, aby zilustrować problem).

Pytanie brzmi - zasugeruj alternatywny sposób (najlepiej - oparty na dostarczonym kodzie źródłowym), aby wykonać dokładnie tę samą funkcję, nie powodując przecieku pamięci w Chrome.

PS 1:

W chromie znalazłem następujące 2 powiązane błędy, które NIE zostały naprawione (dowody - mój kod wciąż przecieka ...):

„manipulowanie img src przez javascript spowoduje ogromny wyciek pamięci” -https://code.google.com/p/chromium/issues/detail?id=36142„Wykorzystanie pamięci rośnie nieskończenie podczas zmiany img.src” -https://code.google.com/p/chromium/issues/detail?id=114570

PS 2:

Jestem w pełni świadomy istniejących, podobnych pytań w stackoverflow i podjąłem wiele prób, ale żadna z nich nie pomogła mi rozwiązać mojego problemu:

Szybkie aktualizowanie obrazu za pomocą Data URI powoduje buforowanie, wyciek pamięciCanvas nawet Img jedzenia RAM i CPUOdśwież obraz za pomocą nowego w tym samym adresie URLUstawienie img.src na daneUrl Pamięć przeciekówPrzeciek pamięci podczas ładowania obrazów za pomocą ustawień javascript

Niektóre próby, na przykład:

Aby mieć pewność, że pamięć podręczna nie jest przyczyną, pracuję w trybie incognito chrome, więc pamięć podręczna nie jest tutaj istotna.Zamiast ustawiać tablicę bajtów jako src, próbowałem użyć adresów URL typu blob (ale nadal występuje podobny przeciek):img.src = window.URL.createObjectURL (nowy Blob ([bytes.buffer], {type: "image / jpeg"}));Próbowałem umieścić obraz w ramce iframe i przeładować go co X klatek: częściowo to pomaga, ale praktycznie nie mogę użyć tego „obejścia”.

* AKTUALIZACJA 29 / Jan *

Zastąpiłem następujące wiersze:

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

Z:

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

i przeciek jest taki sam.

=> Więc mam tylko „1” błąd, który wymaga obejścia (w 2 miejscach): wyciek przy edycji src obrazu.

questionAnswers(1)

yourAnswerToTheQuestion