As we have seen in earlier post first method to reduce memory was reducing font loading. This is described here.
Next method is to use weak references.
2. Making use of weak references
Our application uses observer model to notify data availability. Here a EventNotifier class is used to register listeners for the event. As soon as the event occurs, it fires the event with EventId and EventObject.
Some activities use data that is fetched from server. In order to get this event activity registers itself with the EventNotifier. There is a communicator thread, which downloads data, and calls method public int eventNotify(int eventType, Object eventObject) of EventNotifier. Now this method finds out registered listeners and informs them about update.
Problem comes when Android decides to kill activity for reusing resources. In this case OnDestroy() method may not get called. OnDestroy() handles unregistration activity from EventNotifier. Now as the EventNotifier still holds reference to this activity, it leaks the large amount of memory required by activity, views shown in it, etc.
This can be resolved by marking the ativity reference as weakreference.
WeakReference allows GC to garbage collect activity instance even if, EventNotifier holds WeakReference of it. With strong reference (created by code "EventListener eventListener = this") GC can not garbage collect the activity instance.
Here handler action is created as inner class. Each inner class in Java holds reference to outer class. This handler instance holds instance of Activity which leaks memory. To avoid this make static final instance of handler.
Next method is to use weak references.
2. Making use of weak references
Our application uses observer model to notify data availability. Here a EventNotifier class is used to register listeners for the event. As soon as the event occurs, it fires the event with EventId and EventObject.
Some activities use data that is fetched from server. In order to get this event activity registers itself with the EventNotifier. There is a communicator thread, which downloads data, and calls method public int eventNotify(int eventType, Object eventObject) of EventNotifier. Now this method finds out registered listeners and informs them about update.
Problem comes when Android decides to kill activity for reusing resources. In this case OnDestroy() method may not get called. OnDestroy() handles unregistration activity from EventNotifier. Now as the EventNotifier still holds reference to this activity, it leaks the large amount of memory required by activity, views shown in it, etc.
This can be resolved by marking the ativity reference as weakreference.
public void registerListener(WeakReference < IEventListener > eventListener, int listenerPriority)
WeakReference allows GC to garbage collect activity instance even if, EventNotifier holds WeakReference of it. With strong reference (created by code "EventListener eventListener = this") GC can not garbage collect the activity instance.
3. Making handlers static and final
When a handler is instantiated inside an activity lifecycle, it holds reference to entire activity. If the activity is destroyed but the handler is still in use, GC can not collect the activity as handler is still holding the reference to it. e.g.
protected void onCreate(Bundle savedInstanceState) { ... Handler handler = new Handler(); handler.post(new Runnable() { @ Override public void run() { // some stuff running on UI thread } }); ... }
Here handler action is created as inner class. Each inner class in Java holds reference to outer class. This handler instance holds instance of Activity which leaks memory. To avoid this make static final instance of handler.
private final MyHandler mHandler = new MyHandler(this); /** * Instances of anonymous classes do not hold an implicit * reference to their outer class when they are "static". */ private static final Runnable sRunnable = new Runnable() {@ Override public void run() {} }; @ Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // Post a message and delay its execution for 10 minutes. mHandler.postDelayed(sRunnable, 60 * 10 * 1000); }
No comments:
Post a Comment