When to unregister BroadcastReceiver? In onPause(), onDestroy(), or onStop()?
Solution 1
it depends on where you have register the receiver. The complementary method pairs are
onCreate - onDestroy
onResume - onPause
onStart - onStop
if you register the receiver in the first one then unregister it in it's ending method.
Solution 2
From the Android documentation:
You should implement onStop() to release activity resources such as a network connection or to unregister broadcast receivers.
Then, I would follow these pairs (using @StinePike's analogy):
onResume - onPause
onStart - onStop
Because of the Android Lifecycle, and as @w3bshark mentioned:
In post-HoneyComb (3.0+) devices, onStop() is the last guaranteed handler.
Solution 3
It is just as simple as that, if you want to listen for events even when your activity is not visible then call unregister in onStop() (E.g From Activity A you open Activity B but if you want A to still listening for the events).
But when you only want to listen only for events when your activity is visible then in onPause call unregister() (E.g From Activity A you opened Activity B but now you do not want to listen for events in activity A).
Hope this helps your problem.
Solution 4
An broadcast receiver is an invisible component. All it does it respond to some kind of an change via onReceive() callback.
So it makes sense to activate them , only when your activity is in a state to respond or when it is becoming Enabled /active - which is when onResume() is called.
So it is a better practice to register in onResume() - When activity is visible & Enabled and unregister in onStop() when activity is no longer Active.
Muhammad
Updated on October 18, 2020Comments
-
Muhammad over 3 years
When should I use unregisterReceiver? In
onPause()
,onDestroy()
, oronStop()
?Note: I need the service to run in the background.
Update:
I get an exception releasing receivers
null
.Activity has leaked intent receivers are you missing call to
unregisterReceiver();
Please tell me if there's something wrong, here's my code:
private boolean processedObstacleReceiverStarted; private boolean mainNotificationReceiverStarted; protected void onResume() { super.onResume(); try { registerReceivers(); } catch (Exception e) { Log.e(MatabbatManager.TAG, "MAINActivity: could not register receiver for Matanbbat Action " + e.getMessage()); } } private void registerReceivers() { if (!mainNotificationReceiverStarted) { mainNotificationReceiver = new MainNotificationReceiver(); IntentFilter notificationIntent = new IntentFilter(); notificationIntent .addAction(MatabbatManager.MATABAT_LOCATION_ACTION); notificationIntent .addAction(MatabbatManager.MATABAT_New_DATA_RECEIVED); notificationIntent .addAction(MatabbatManager.STATUS_NOTIFCATION_ACTION); registerReceiver(mainNotificationReceiver, notificationIntent); mainNotificationReceiverStarted = true; } if (!processedObstacleReceiverStarted) { processedObstacleReceiver = new ProcessedObstacleReceiver(); registerReceiver(processedObstacleReceiver, new IntentFilter( MatabbatManager.MATABAT_ALARM_LOCATION_ACTION)); processedObstacleReceiverStarted = true; } } private void unRegisterReceivers() { if (mainNotificationReceiverStarted) { unregisterReceiver(mainNotificationReceiver); mainNotificationReceiverStarted = false; } if (processedObstacleReceiverStarted) { unregisterReceiver(processedObstacleReceiver); processedObstacleReceiverStarted = false; } } @Override protected void onDestroy() { // TODO Auto-generated method stub super.onDestroy(); try { unRegisterReceivers(); mWakeLock.release();//keep screen on } catch (Exception e) { Log.e(MatabbatManager.TAG, getClass() + " Releasing receivers-" + e.getMessage()); } }
-
stinepike almost 9 years@nAkhmedov, can you please explain
-
Brian over 8 yearsThis often won't work, since onStart/onResume of activity B happens before onStop of activity A (when moving from A -> B). The wrong activity may catch your broadcast, causing difficult to debug issues.
-
w3bshark over 8 yearsThe last lifecycle event handler that is guaranteed to be called before an app is terminated (if you’re supporting pre-HoneyComb devices) is onPause. If you're only supporting Post-HoneyComb devices, then onStop is the last guaranteed handler. You should never assume onDestroy will be called and thus, you should unregister the receiver before this lifecycle event. Source: Android Developer docs
-
Kevin Coppock about 8 years@w3bshark: If your process is being killed to reclaim memory, then it doesn't matter if you unregister the receiver, as your application will be evicted from memory (including your receiver). You only need to worry about killable states if you have something that's persistent and you must guarantee the method will be called.
-
Pedro Varela about 8 yearsThus, you have to register and unregister receivers in onResume and onPause, because they will be call for sure before the fragment or the activity is destroyed.
-
Bawa over 4 yearswhen we call activity A to activity B, activity A goes in onStop state and the receiver will get unregistered and won't listen to broadcasts.