I am working on an application that is creating and removing a lot of DOMs. I've notice that the process memory from the browser tab continuously increases, despite the javascript heap memory remaining constant. In a test application I create and remove divs from a parent div.
http://jsfiddle.net/PSxPz/2/
<button onclick="createStuff()">Create</button>
<button onclick="deleteStuff()">Delete</button>
<div id="parent"></div>
function createStuff() {
    var parentDiv = document.getElementById('parent');
    for (var i = 0; i < 50000; i++) {
        var child = document.createElement('div');
        child.id = i;
        child.textContent = i;
        parentDiv.appendChild(child);
        child = null;
    }
    parentDiv = null;
}
function deleteStuff() {
    var parentDiv = document.getElementById('parent');
    for (var i = 0; i < 50000; i++) {
        var child = document.getElementById(i);
        parentDiv.removeChild(child);
        child = null;
    }
    parentDiv = null;
}
I've confirmed that the javascript heap is not leaking with the chrome dev tools (I'm new to them so I could have missed something). However the memory for the process continues to increase. From everything I've read I suspect that the removed doms are still in the dom heap.
Other posts also say that the browser will eventually free the memory allocated to the removed doms. In the above jsfiddle example I've hit create and delete several times. My javascript heap is steady at 4.9MB. My process memory is up to 115MB. I've waited 30 mins and it hasn't gone down at all.
Questions
Thanks for the help!
Edit
I have used the chrome dev tools and the javascript heap is not growing. Interestingly, the only thing that changes between the heap snapshots is an (array) object. It's my understanding that anything in parenthesis is controlled by the browser and outside of my reach. Each subsequent create->delete removes the old (array) object and creates a new one during the delete.
In timeline I can see that the javascript heap is constant and the nodes get cleaned up, but the memory as shown with (shift + esc) never goes down even after the node count drops.


It seems like I'm doing everything I can to make sure I cleanup my javascript heap, but the dom cleanup is out my reach and independent of the javascript GC. Is this statement correct?
Are the removed doms part of the young generation heap? Is there a way to set a limit on this heap size? I repeated the test until I had reached 500MB and still no cleanup. I'm using Chrome 35.0.1916.114 btw.
As a solution, try to nullify these variables after use, or add use strict to enable a stricter mode of JavaScript that prevents accidental global variables.
Detached DOM elements are the elements which have been removed from the DOM but their memory is still retained because of JavaScript. This means that as long the element have a reference to any variable or an object anywhere, it does not garbage collected even after destroyed from the DOM.
JavaScript code can experience memory leaks by keeping hidden references to objects. Hidden references can cause memory leaks in many unexpected ways.
I know you asked about Chrome, but I'll describe how it works in Firefox, hoping it might be of interest to you and other readers. Chrome may work similarly, I'm not sure.
With your testcase the Firefox memory usage does not continuously increase. Only the first time you create/remove the elements does the memory usage increase permanently. During subsequent create/remove cycles, all allocated memory is subsequently reclaimed.
At least in Firefox you can't force this memory to be deallocated without reloading the page. If you really need to allocate this much memory temporarily, you should do this in an iframe you can throw away when you're done.
Technical details follow:
In Firefox there's a tool to check the memory usage called about:memory. It breaks up the used memory by category and has controls to force-clean memory (GC/CC/Minimize memory).
Here's how the relevant bit of about:memory looks after you create/remove the DOM elements and after GC kicks in:
├──34.34 MB (03.30%) -- window(http://fiddle.jshell.net/PSxPz/2/show/)
│  ├──26.54 MB (02.55%) -- layout
│  │  ├──13.95 MB (01.34%) -- (8 tiny)
│  │  │  ├───7.63 MB (00.73%) ── line-boxes
│  │  │  ├───4.00 MB (00.38%) ── pres-contexts
│  │  │  ├───2.26 MB (00.22%) ── pres-shell
│  │  │  ├───0.04 MB (00.00%) ── style-structs
│  │  │  ├───0.01 MB (00.00%) ── rule-nodes
│  │  │  ├───0.01 MB (00.00%) ── style-contexts
│  │  │  ├───0.00 MB (00.00%) ── style-sets
│  │  │  └───0.00 MB (00.00%) ── text-runs
│  │  └──12.59 MB (01.21%) -- frames
│  │     ├───7.25 MB (00.70%) ── nsBlockFrame
│  │     ├───5.34 MB (00.51%) ── nsTextFrame
│  │     └───0.00 MB (00.00%) ── sundries
│  └───7.80 MB (00.75%) -- (4 tiny)
│      ├──7.51 MB (00.72%) ++ dom
│      ├──0.29 MB (00.03%) ++ js-compartment(http://fiddle.jshell.net/PSxPz/2/show/)
│      ├──0.00 MB (00.00%) ── style-sheets
│      └──0.00 MB (00.00%) ── property-tables
(If the DOM nodes were already removed from the document, but not garbage collected yet, they show up under orphan-nodes measurement.)
Most of the additional memory (requested when the DOM nodes were created) is reserved for layout.
This behavior is based on measurements that showed that real-life web pages usually need around the same number of layout objects during their lifetime: a typical web page doesn't allocate 10,000s of frames only to destroy them and show a very simple page instead, like the testcase here does.
This memory management behavior pays off in improved speed of frame allocation/deallocation, reduced memory fragmentation, and avoiding nasty security bugs when a frame is accessed after it's destroyed.
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