Avoid controlled memory leaks
Memory leakage is the focus of memory optimization. How to avoid, find and solve memory leakage is very important
What is a memory leak
Every application needs memory to do its job, and to ensure that every Android application has enough memory, the Android system needs to manage memory allocation effectively. The GC is triggered when Android runs out of memory, and the GC uses the root search algorithm. A memory leak is when a useless object is reachable from GC Roots, causing the GC to be unable to reclaim the object. There are three main causes of memory leaks in general
- Memory leaks caused by developers coding themselves
- Leakage caused by the tripartite framework
- Leakage caused by Android system or third party ROM
In general, the second and third are uncontrollable, but the first is. Here are some examples of common memory leak scenarios
2. Memory leakage scenario
A static instance of a non-static inner class
A non-static inner class holds a reference to an instance of an external class. If an instance of a non-static inner class is static, it indirectly holds a reference to an instance of an external class for a long time, preventing it from being reclaimed by the system
public class MainActivity extends AppCompatActivity { private static Object inner; private Button button; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); button = findViewById(R.id.button_test); button.setOnClickListener(v -> { createInnerClass(); finish(); }); } void createInnerClass(){ class InnerClass{ } inner = new InnerClass(); }}Copy the code
When clicked on a Button, a static instance of InnerClass, a non-static inner class, is created at new InnerClass(). The inner instance lives as long as the application and holds a reference to SecondActivity. SecondActivity cannot be recycled
Multithreaded related anonymous inner class/non-static inner class
Anonymous inner classes also hold references to instances of external classes. Multithreaded related classes include AsyncTask, Thrad, and classes that implement Runnable interfaces. Their anonymous inner classes/non-static inner classes will cause memory leaks if they do time-consuming operations
public class AsyncTaskActivity extends AppCompatActivity { private Button button; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_async_task); button = findViewById(R.id.button_async); button.setOnClickListener(v -> { startAsyncTask(); finish(); }); } void startAsyncTask(){ new AsyncTask<Void, Void, Void>(){ @Override protected Void doInBackground(Void... voids) { while(true); } }.execute(); }}Copy the code
An AsyncTask is instantiated in the startAsyncTask method. While the asynchronous task is performing a time-consuming operation in the background, the entire Activity is destroyed and the Activity instance held by the AsyncTask is not garbage collected until the asynchronous task is finished. Similarly, if a custom AsyncTask is a non-static inner class, memory leaks will also occur. The solution is to define a custom static AsyncTask as follows
public class AsyncTaskActivity extends AppCompatActivity { private Button button; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_async_task); button = findViewById(R.id.button_async); button.setOnClickListener(v -> { startAsyncTask(); finish(); }); } void startAsyncTask(){ new MyAsyncTask().execute(); } private static class MyAsyncTask extends AsyncTask<Void, Void, Void>{ @Override protected Void doInBackground(Void... voids) { while (true); }}}Copy the code
Handler memory leak
Handler messages are stored in a MessageQueue. Some messages cannot be processed immediately. They are stored in a MessageQueue for a long time. Handler also causes activities or services that reference it to not be recycled
public class HandlerActivity extends AppCompatActivity { private Button button; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_handler); button = findViewById(R.id.button_handler); final Handler handler = new Handler(){ @Override public void handleMessage(@NonNull Message msg) { super.handleMessage(msg); }}; button.setOnClickListener(v -> { handler.sendMessageDelayed(Message.obtain(), 6000); finish(); }); }}Copy the code
Handler is an instance of a non-static anonymous inner class that implicitly refers to the external class HandlerActivity. In this case, when we click on a button, the HandlerActivity ends, but the message in the Handler hasn’t been processed yet. Therefore HandlerActivity cannot be recycled. There are two solutions. One is to use static Handler inner classes, which hold objects with weak references, as shown below
public class HandlerActivity extends AppCompatActivity { private Button button; private MyHandler myHandler = new MyHandler(this); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_handler); button = findViewById(R.id.button_handler); button.setOnClickListener(v -> { myHandler.sendMessageDelayed(Message.obtain(), 6000); finish(); }); } public void show(){} private static class MyHandler extends Handler{ private final WeakReference<HandlerActivity> activityWeakReference; public MyHandler(HandlerActivity activity){ activityWeakReference = new WeakReference<>(activity); } @Override public void handleMessage(Message message){ if(activityWeakReference ! = null && activityWeakReference.get() == null){ activityWeakReference.get().show(); }}}}Copy the code
MyHandler is a static inner class that holds HandlerActivity objects that use weak references to avoid memory leaks.
Another solution is to remove messages from MessageQueue in the onDestry lifecycle callback, as shown below
@Override public void onDestroy(){ if(myHandler ! = null){ myHandler.removeCallbacksAndMessages(null); } super.onDestroy(); }Copy the code
Cleaning up Callbacks and Messages on the point of destruction may not completely clean up Messages in the Handler, so the first approach is recommended
Context is not used correctly
In cases where the Activity Context is not required, for example, the Dialog Context must use the Activity Context. Consider using an Application Context instead of an Activity Context to avoid Activity leaks, such as the following singleton pattern
public class AppSettings { private Context appContext; private static AppSettings appSettings = new AppSettings(); public static AppSettings getInstance(){ return appSettings; } public final void setup(Context context){ appContext = context; }}Copy the code
AppSettings, as a static object, has a lifetime longer than the Activity. When the screen rotates, by default, the system destroys the current Activity because the current Activity calls the setup method and passes in the Activity Context, making the Activity held by a singleton that can’t be collected by the garbage collector, resulting in a memory leak. The solution is to use the Application Context as follows
public final void setup(Context context){
appContext = context.getApplicationContext();
}
Copy the code
Static View
Using a static View prevents you from reading and rendering the View every time you start an Activity. However, a static View holds a reference to the Activity and cannot be recycled. The solution is to set the static View to NULL in onDestroy
public class ViewActivity extends AppCompatActivity { private static Button button; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_view); button = findViewById(R.id.button_view); button.setOnClickListener(v -> { finish(); }); } @Override protected void onDestroy() { button = null; super.onDestroy(); }}Copy the code
WebView
WebView will be different in different Android versions, and the WebView of ROM customized by different manufacturers will also be different, which leads to great compatibility problems of WebView. WebView will always have memory leakage problem, and the memory will not be released when the WebView is used once in the application. The usual solution is to open a single process for the WebView, using AIDL to communicate with the main application process, and the WebView process can be destroyed at the appropriate time according to the business needs
The resource object is not closed
Resource objects such as Cursor and File are often buffered, causing memory leaks. Therefore, when a resource object is not in use, be sure to close its references and set them to NULL, usually ina finally statement
Objects in the collection are not cleaned up
It is common to add a reference to an object to a collection, and when the object is no longer needed, the collection gets bigger and bigger if the reference is not removed from the collection. The situation is even worse if the collection is static
BitMap object
After a temporarily created relatively large bitmap object is transformed into a new bitmap object, the original bitmap should be reclaimed as soon as possible, so that the space occupied by the original bitmap can be released faster. Avoid static variables holding large Bitmaps or other large data objects, and if they do, empty the static variable as soon as possible
The listener is not closed
Many system services (such as TelephonyMannager, SensorManager) require registers and unregister listeners, and we need to make sure that we unregister those listeners when appropriate. Add the Listener manually and remember to remove it when appropriate
Memory Monitor
This is a memory analysis tool built into AS, opened in the Profile below the editor. For example, if we run any of the code in the example above, we can see a memory graph similar to the following
The main functions of this tool are as follows
- Displays graphs of available and allocated Java memory in real time
- Display garbage collection events in real time
- Start the garbage collection event
- Quickly test whether the slowness of the application is related to excessive garbage collection events
- Quickly test whether an application crash is related to running out of memory
1. Large memory application and GC
As can be seen from the figure above, the allocated memory increases sharply, which is the scenario of large memory allocation. We need to determine whether the memory cost is reasonable, and optimize this big data to reduce performance loss. This is followed by a sharp drop, which represents a garbage collection event used to free memory
2. Memory jitter
Memory jitter generally refers to multiple memory allocations and releases in a short period of time. Severe memory jitter may cause application delays. The main cause of memory jitter is the frequent creation of objects in a short period of time (possibly in a loop), and the memory will also perform GC frequently to cope with this situation. While non-parallel GC is in progress, all other threads are suspended, waiting for the GC operation to complete and resume work. If GC is frequent, it will produce a lot of pause time, which will reduce the interface drawing time, so that the time to draw a frame many times is more than 16ms, resulting in the phenomenon of interface lag. This adds up to memory jitter, which produces a jagged jitter graph
LeakCanary
MAT, an Eclipse-based plug-in, can be used to analyze memory problems, but it is difficult and not very efficient. For a memory leak problem, it may require multiple checks and comparisons. Square has opened source LeakCanary based on MAT to make it easier and faster to detect memory leaks
1. Use LeakCanary
First configure build.gradle
Dependencies {debugImplementation 'com. Squareup. Leakcanary: leakcanary - android: 1.6.3' releaseImplementation 'com. Squareup. Leakcanary: leakcanary - android - no - op: 1.6.3'}Copy the code
Next add the following code to Application
public class LeakApplication extends Application { @Override public void onCreate() { super.onCreate(); if (LeakCanary.isInAnalyzerProcess(this)) { // This process is dedicated to LeakCanary for heap analysis. // You should not init your app in this process. return; } LeakCanary.install(this); }}Copy the code
The code in the if condition is used to filter and return if the current process is used to do heap analysis for LeakCanary, otherwise the install method for LeakCanary will be executed. This allows us to use LeakCanary, which will prompt if it detects a memory leak in an Activity
2. LeakCanary Application example
The above example can only detect memory leaks for activities, but if we need to detect memory leaks for other classes, we need RefWatcher to monitor them. Rewrite Application as follows
public class LeakApplication extends Application { private RefWatcher refWatcher; @Override public void onCreate() { super.onCreate(); refWatcher = setupLeakCanary(); } private RefWatcher setupLeakCanary(){ if(LeakCanary.isInAnalyzerProcess(this)){ return RefWatcher.DISABLED; } return LeakCanary.install(this); } public static RefWatcher getRefWatcher(Context context){ LeakApplication leakApplication = (LeakApplication) context.getApplicationContext(); return leakApplication.refWatcher; }}Copy the code
The Install method returns the RefWatcher to monitor the object, and LeakApplication also provides the getRefWatcher static method to return the global RefWatcher. Finally, for an example, we introduce LeakCanary monitoring in a piece of code with a memory leak, as follows
public class LeakActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_leak); LeakThread leakThread = new LeakThread(); leakThread.start(); } class LeakThread extends Thread{ @Override public void run() { try { Thread.sleep(6 * 60 * 1000); } catch (InterruptedException e) { e.printStackTrace(); } } } @Override protected void onDestroy() { super.onDestroy(); RefWatcher refWatcher = LeakApplication.getRefWatcher(this); refWatcher.watch(this); }}Copy the code
LeakActivity has a memory leak because the non-static inner class LeakThread holds a reference to the external class, and time-consuming operations in LeakThread cause the Activity to fail to be released. After getting the refWatcher object, call the Watch method to observe the object to monitor, this. Of course, the onDestroy method is redundant in this demo, as LeakCanary launches an ActivityRefWatcher class after calling the Install method, which automatically monitors the Activity for memory leaks after executing the onDestroy method. For example, if you want to monitor your Fragment, add the onDestroy method to the Fragment. The app is then run to generate Leaks app ICONS on the interface. By constantly switching between destroying and creating activities on the vertical screen, a prompt box will pop up, and memory leaks will be displayed via Notification, and clicking will take you to the memory leak details page, which will display a specific reference chain
The inner class LeakThread of MainActiviy refers to LeakThread’s This $0. This $0 means that the inner class automatically retains a reference to the external class where the MainActiviy instance is located, as shown in the last line of the detail. This will cause MainActivity to fail to be GC, resulting in a memory leak
Of course, the solution is simply to declare the inner class static and not give a hint of a memory leak