Record heap snapshots

Meggin Kearney
Meggin Kearney
Sofia Emelianova
Sofia Emelianova

Learn how to record heap snapshots with Memory > Profiles > Heap snapshot and find memory leaks.

The heap profiler shows memory distribution by your page's JavaScript objects and related DOM nodes. Use it to take JS heap snapshots, analyze memory graphs, compare snapshots, and find memory leaks. For more information, see Objects retaining tree.

Take a snapshot

To take a heap snapshot:

  1. On a page you want to profile, open DevTools and navigate to the Memory panel.
  2. Select the Heap snapshot profiling type, then select a JavaScript VM instance, and click Take snapshot.

A selected profiling type and JavaScript VM instance.

When the Memory panel loads and parses the snapshot, it shows the total size of reachable JavaScript objects below the snapshot title in the HEAP SNAPSHOTS section.

The total size of reachable objects.

Snapshots show only the objects from the memory graph that are reachable from the global object. Taking a snapshot always starts with garbage collection.

A heap snapshot of scattered Item objects.

Clear snapshots

To remove all snapshots, click Clear all profiles:

Clear all profiles.

View snapshots

To inspect snapshots from different perspectives for different purposes, select one of the views from the drop-down menu at the top:

View Content Purpose
Summary Objects grouped by constructor names and sources. Use it to hunt down objects and their memory use based on type. Helpful for tracking DOM leaks.
Comparison Differences between two snapshots. Use it to compare two (or more) snapshots, before and after an operation. Confirm the presence and cause of a memory leak by inspecting the delta in freed memory and reference count.
Containment Heap contents Provides a better view of object structure, and helps analyze objects referenced in the global namespace (window) to find what keeps them around. Use it to analyze closures and dive into your objects at a low level.
Statistics Pie chart of memory allocation See the realtive sizes of memory parts allocated to code, strings, JS arrays, typed arrays, and system objects.

The Summary view selected from the drop-down menu at the top.

Summary view

Initially, a heap snapshot opens in the Summary view that lists Constructors in a column. Constructors are named after the JavaScript function that created the object, plain objects names are based on the proper they contain, and some names are special entries. All objects are grouped by their names first, then by the line in the source file they come from, for example, source-file.js:line-number.

You can expand grouped constructors to see the objects they instantiated.

The Summary view with an expanded constructor.

To filter out irrelevant constructors, type a name that you want to inspect in the Class filter at the top of the Summary view.

The numbers next to constructor names indicate the total number of objects created with the constructor. The Summary view also shows the following columns:

  • Distance shows the distance to the root using the shortest simple path of nodes.
  • Shallow size shows the sum of shallow sizes of all objects created by a certain constructor. The shallow size is the size of memory held by an object itself. Generally, arrays and strings have larger shallow sizes. See also Object sizes.
  • Retained size shows the maximum retained size among the same set of objects. Retained size is the size of memory that you can free by deleting an object and making its dependents no longer reachable. See also Object sizes.

When you expand a constructor, the Summary view shows you all of its instances. Each instance gets a breakdown of its shallow and retained sizes in the corresponding columns. The number after the @ character is the object's unique ID. It lets you compare heap snapshots on per-object basis.

Constructor filters

Summary view lets you filter constructors based on common cases of inefficient memory usage.

To use these filters, select one of the following options from the rightmost drop-down menu in the action bar:

  • All objects: all objects captured by the current snapshot. Set by default.
  • Objects allocated before snapshot 1: objects that were created and remained in memory before the first snapshot was taken.
  • Objects allocated between Snapshots 1 and Snapshots 2: view the difference in objects between the most recent snapshot and the previous snapshot. Each new snapshot adds an increment of this filter to the drop-down list.
  • Duplicated strings: string values that have been stored multiple times in memory.
  • Objects retained by detached nodes: objects that are kept alive because a detached DOM node references them.
  • Objects retained by the DevTools console: objects kept in memory because they were evaluated or interacted with through the DevTools console.

Special entries in Summary

In addition to grouping by constructors, the Summary view also groups objects by:

  • Built-in functions such as Array or Object.
  • HTML elements grouped by their tags, for example, <div>, <a>, <img>, and others.
  • Functions you defined in your code.
  • Special categories that aren't based on constructors.

Constructor entries.

(array)

This category includes various internal array-like objects that don't directly correspond to objects visible in JavaScript.

For example, the contents of JavaScript Array objects are stored in a secondary internal object named (object elements)[], to allow easier resizing. Similarly, the named properties in JavaScript objects are often stored in secondary internal objects named (object properties)[] that are also listed in the (array) category.

(compiled code)

This category includes internal data that V8 needs so that it can run functions defined by JavaScript or WebAssembly. Each function can be represented in a variety of ways, from small and slow to large and fast.

V8 automatically manages memory usage in this category. If a function runs many times, V8 uses more memory for that function so that it can run faster. If a function hasn't run in a while, V8 may clear the internal data for that function.

(concatenated string)

When V8 concatenates two strings, such as with the JavaScript + operator, it may choose to represent the result internally as a "concatenated string" also known as the Rope data structure.

Rather than copying all of the characters of the two source strings into a new string, V8 allocates a small object with internal fields called first and second, which point to the two source strings. This lets V8 save time and memory. From the perspective of JavaScript code, these are just normal strings, and they behave like any other string.

InternalNode

This category represents objects allocated outside V8, such as C++ objects defined by Blink.

To see C++ class names, use Chrome for Testing and do the following:

  1. Open DevTools and turn on Settings > Experiments > Show option to expose internals in heap snapshots.
  2. Open the Memory panel, select Heap snapshot, and turn on Expose internals (includes additional implementation-specific details).
  3. Reproduce the issue that caused the InternalNode to retain a lot of memory.
  4. Take a heap snapshot. In this snapshot, objects have C++ class names instead of InternalNode.
(object shape)

As described in Fast Properties in V8, V8 tracks hidden classes (or shapes) so that multiple objects with the same properties in the same order can be represented efficiently. This category contains those hidden classes, called system / Map (unrelated to JavaScript Map), and related data.

(sliced string)

When V8 needs to take a substring, such as when JavaScript code calls String.prototype.substring(), V8 may choose to allocate a sliced string object rather than copying all of the relevant characters from the original string. This new object contains a pointer to the original string and describes which range of characters from the original string to use.

From the perspective of JavaScript code, these are just normal strings, and they behave like any other string. If a sliced string is retaining a lot of memory, then the program may have triggered Issue 2869 and might benefit from taking deliberate steps to "flatten" the sliced string.

system / Context

Internal objects of type system / Context contain local variables from a closure—a JavaScript scope that a nested function can access.

Every function instance contains an internal pointer to the Context in which it executes, so that it can access those variables. Even though Context objects aren't directly visible from JavaScript, you do have direct control over them.

(system)

This category contains various internal objects that haven't (yet) been categorized in any more meaningful way.

Comparison view

The Comparison view lets you find leaked objects by comparing multiple snapshots to each other. For example, doing an action and reversing it, like opening a document and closing it, shouldn't leave extra objects behind.

To verify that a certain operation doesn't create leaks:

  1. Take a heap snapshot before performing an operation.
  2. Perform an operation. That is, interact with a page in some way that you think might be causing a leak.
  3. Perform a reverse operation. That is, do the opposite interaction and repeat it a few times.
  4. Take a second heap snapshot and change its view to Comparison, comparing it to Snapshot 1.

The Comparison view shows the difference between two snapshots. When expanding a total entry, added and deleted object instances are shown:

Comparing to Snapshot 1.

Containment view

The Containment view is a "bird's eye view" of your application's objects structure. It lets you peek inside function closures, observe VM internal objects that together make up your JavaScript objects, and to understand how much memory your application uses at a very low level.

The view provides several entry points:

  • DOMWindow objects. Global objects for JavaScript code.
  • GC roots. GC roots used by the VM's garbage collector. GC roots can consist of built-in object maps, symbol tables, VM thread stacks, compilation caches, handle scopes, and global handles.
  • Native objects. Browser objects "pushed" inside the JavaScript virtual machine to allow automation, for example, DOM nodes and CSS rules.

The Containment view.

The Retainers section

The Retainers section at the bottom of the Memory panel shows objects that point to the object selected in the view. The Memory panel updates the Retainers section when you select a different objects in any of the views except Statistics.

The Retainers section.

In this example, the selected string is retained by the x property of an Item instance.

Ignore retainers

You can hide retainers to find out of any other objects retain the selected one. With this option, you don't have to first remove this retainer from the code and then retake the heap snapshot.

The 'Ignore this retainer' option in the drop-down menu.

To hide a retainer, right-click and select Ignore this retainer. Ignored retainers are marked as ignored in the Distance column. To stop ignoring all retainers, click Restore ignored retainers in the action bar at the top.

Find a specific object

To find an object in the collected heap you can search using Ctrl + F and enter the object ID.

Name functions to distinguish closures

It helps a lot to name the functions so you can distinguish between closures in the snapshot.

For example, the following code doesn't use named functions:

function createLargeClosure() {
  var largeStr = new Array(1000000).join('x');

  var lC = function() { // this is NOT a named function
    return largeStr;
  };

  return lC;
}

Whilst this example does:

function createLargeClosure() {
  var largeStr = new Array(1000000).join('x');

  var lC = function lC() { // this IS a named function
    return largeStr;
  };

  return lC;
}

Named function in a closure.

Uncover DOM leaks

The heap profiler has the ability to reflect bidirectional dependencies between browser-native objects (DOM nodes and CSS rules) and JavaScript objects. This helps discover otherwise invisible leaks happening due to forgotten detached DOM subtrees floating around.

DOM leaks can be bigger than you think. Consider the following example. When is the #tree garbage collected?

  var select = document.querySelector;
  var treeRef = select("#tree");
  var leafRef = select("#leaf");
  var body = select("body");

  body.removeChild(treeRef);

  //#tree can't be GC yet due to treeRef
  treeRef = null;

  //#tree can't be GC yet due to indirect
  //reference from leafRef

  leafRef = null;
  //#NOW #tree can be garbage collected

#leaf maintains a reference to its parent (parentNode) and recursively up to #tree, so only when leafRef is nullified is the whole tree under #tree a candidate for GC.

DOM subtrees