Coming soon! See flutter/devtools#3951.
The text below is under construction.
This page describes motivation behind memory leak tracking. Read more about leak tracking in overview.
In most cases, an object's lifecycle is managed well by the Dart garbage collector.
However, some objects (memory-risky objects) require special care to avoid the risk of memory leaks:
- Objects that allocate significant memory for their members
- Objects that reference a significant number of other objects, designed to be short living (like the entire widget tree), and therefore prevent them from being GCed in time.
Such objects usually have a method dispose
, which has two purposes:
- Releases the resources, to make sure they do not stay longer than needed.
- Explicitly marks the object as not needed, to make lifecycle management easier. For example:
a. Some objects restrict invocation of some methods after disposal.
b. It is expected that objects should be available for GC after disposal, so owners of the objects should null the references after invocation of
dispose
.
To ensure memory efficiency of applications, engineers should watch that:
- All disposables are disposed when they are not needed any more.
- Large objects and objects that reference large sets of other objects are not reachable after usage.
- Objects allocate a reasonable amount of memory necessary to perform their tasks.
Engineers want their applications to be memory efficient, but they want to minimize time and mental effort spent worrying about the items above.
leak_tracker fully automates item #1, and almost* fully automates item #2.
Note
*If a large set of objects is held from GC longer than needed, some of them are likely to be disposables, and thus leaks will be detected by the leak_tracker. This is not always true, though; in rare cases, a large set of objects is held from GC, but the set does not contain any disposables tracked by leak_tracker.
As a result, leak_tracker
increases confidence and reduces engineering effort related to memory efficiency.
Item #3 is not covered by leak_tracker
. There are no known tools to automate it so far. All known methods are manual:
- Analyzing the consumed memory using DevTools snapshotting and diffing
- Memory baselining in regression tests
How large can a memory leak be when disposal is forgotten or when an object is referenced after disposal?
The cases below are selected from the GitHub issue history and illustrate three types of memory issues:
- Out of memory crash
- 100K+ leaks
- Leaks with no numbers recorded, but significant enough to file, investigate and fix the issue
- VideoController caused crash: flutter/flutter#72643
- Canvas size is growing infinitely: flutter/flutter#58437
- WebGL caused 2GB+ Leak: flutter/flutter#52485
-
Disposed but not GCed Route is leaking: flutter/flutter#88073. (A route may reference the entire page widget tree.)
-
A Flutter customer building a large application found the application was leaking over 300KB at startup. Many of the leaked objects would be detected by this package.
Issues here are fixed by updating ‘dispose’, not invoking it. That means not invoked ‘dispose’ would cause a significant leak.
Vertices
causes app crash: flutter/flutter#54762VideoPlayerController
causes leak 25MB: flutter/flutter#86477AnimationController
causes significant leak: flutter/flutter#84730ImageShader
causes significant leak: flutter/flutter#82832- Google map causes app crash: flutter/flutter#35243
- Images in
GridView
causes crash: flutter/flutter#19558 CanvasImage
causes significant leak: flutter/flutter#57746