I encountered the following 2 (huge!) memory leaks in Chrome:
Note that in Internet Explorer there is NO memory leak what so ever!
Some background: I'm working on a project where an external camera provides a live feed of images (let's say 100 frames per second).
The main 3 functionalities of the project are:
You are welcome to download the following standalone code (simply save it as "leak.html" and execute it), and see for yourself:
<!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>
This example code takes a static image (of a chair) and modifies it randomly every frame (just to simulate my real camera feed).
For the first 1000 frames it shows the image, and stores the last 10 frames in a cyclic array, and from then on it just shows the 10 frames last recorded (in a loop).
(Obviously my real project is much more complicated, I just simplified it to illustrates the problem).
The question is - please suggest an alternative way (preferably - based on the provided source code) to perform the exact same functionality, without causing a memory leak in Chrome.
PS 1:
In chromium I found the following 2 related bugs, which were NOT really fixed (evidence - my code still leaks...):
PS 2:
I'm fully aware of existing, similar questions in stackoverflow, and I made a lot of attempts, but none of them helped me solve my problem:
Some attempts I made, for example:
* UPDATE 29/Jan *
I replaced the following lines:
var $tmpImage = $(image);
var clonedImage = $tmpImage.clone()[0];
With:
var clonedImage = new Image();
clonedImage.src = newImgSrc;
and the leak is the same.
=> So I am down to 'only' 1 bug that requires a workaround (in 2 places): leak when editing an image's src.
Google Chrome is one of the popular web browsers and Chrome memory leak is one of the common issues. Today we will talk about this problem on the MiniTool website. If you find there are many tabs of Chrome in Task Manager and Chrome uses much memory, follow these solutions below to easily fix the issue.
Visualize memory consumption with the performance profiler The performance profiler in Chrome can visualize memory usage and graph it over time. To try this out, open the DevTools in Chrome and switch to the Performance tab. Note that we use an Incognito window when measuring performance.
I had the same problem. The only workaround I found was to reduce the number of new Image() to use (ideally one):
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 });
Note that images will be loaded in serie. If if want to load 2 images in parallel, you'll need to instantiate 2 ImageLoader.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With