Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Fragment Activity - app died, no saved state

I'm facing a very confusing issue for a very small number of my users. The error occurs when a button is pressed inside a Fragment, that starts another Fragment Activity. Here is the stack trace:

I/20:22:23.901 ActivityManager( 1668)
Start proc com.brandall.nutter for activity com.brandall.nutter/.ActivityHomeFragment: pid=8956 uid=10125 gids={50125, 3003, 3001, 3002, 1015, 1023, 1006, 1028}
I/20:22:23.881 WindowState( 1668)
WIN DEATH: Window{41ed1948 u0 com.brandall.nutter/com.brandall.nutter.ActivityLinkAppsFragment}
W/20:22:23.881 ActivityManager( 1668)
Force removing ActivityRecord{411c4188 u0 com.brandall.nutter/.ActivityLinkAppsFragment}: app died, no saved state
I/20:22:23.881 WindowState( 1668)
WIN DEATH: Window{41b6a178 u0 Toast EXITING}
W/20:22:23.881 InputDispatcher( 1668)
Attempted to unregister already unregistered input channel '41ed1948 com.brandall.nutter/com.brandall.nutter.ActivityLinkAppsFragment (server)'
W/20:22:23.871 ActivityManager( 1668)
Scheduling restart of crashed service com.brandall.nutter/.TTSS in 80000ms
I/20:22:23.871 ActivityManager( 1668)
Process com.brandall.nutter (pid 8907) has died.
I/20:22:23.871 WindowState( 1668)
WIN DEATH: Window{411d4ff0 u0 com.brandall.nutter/com.brandall.nutter.ActivityHomeFragment}
E/20:22:23.871 InputDispatcher( 1668)
channel '41ed1948 com.brandall.nutter/com.brandall.nutter.ActivityLinkAppsFragment (server)' ~ Channel is unrecoverably broken and will be disposed!
W/20:22:23.871 InputDispatcher( 1668)
channel '41ed1948 com.brandall.nutter/com.brandall.nutter.ActivityLinkAppsFragment (server)' ~ Consumer closed input channel or an error occurred.  events=0x9

Due to this line of the error:

com.brandall.nutter/.ActivityLinkAppsFragment}: app died, no saved state

I've been reading many posts about Fragment saved states, but none seem to apply in my circumstances, rather to the Fragments themselves, which aren't mentioned in the stack trace. The other posts suggest adding to each Fragment:

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setRetainInstance(true);
}

I currently don't override the onCreate method in any of my Fragments.

I've also seen suggested adding this to each Fragment:

@Override
public void onSaveInstanceState(Bundle outState) {
    super.onSaveInstanceState(outState);
    setUserVisibleHint(true);
}

Apart from the fact that it's only happening to a very small number of users, the most baffling issue is with the Context used:

final Intent sa = new Intent(getActivity(), ActivityLinkAppsFragment.class);
getActivity().startActivity(sa);

My application has a foreground Service and if I instead use the static Context of that (through a static getServiceContext() method) in the Intent above, the problem no longer occurs for the users and the Fragment Activity opens correctly.

I don't understand why the use of another Context should prevent the issue from happening, but I hope with the information I've supplied, it will make sense to someone!

I can supply the FragmentPagerAdapter code I'm using if it proves to be relevant, but it's pretty standard.

I thank you in advance

EDIT - Something very important I forgot to add. This does not cause the app to crash. Instead the Activity from which the Fragment is in, gets restarted immediately.

ANSWER - This was caused due to me calling System.exit(0) under circumstances I thought only possible when the user wanted to 'flush' the memory usage of the app. I was wrong and it could also be called when the user's device was handling low memory conditions. @beworker's answer below is marked as correct, as he noted ActivityManagerService.handleAppDiedLocked() which was a result of this.

like image 340
brandall Avatar asked Aug 21 '13 18:08

brandall


People also ask

How to restore fragment state?

Various Android system operations can affect the state of your fragment. To ensure the user's state is saved, the Android framework automatically saves and restores the fragments and the back stack. Therefore, you need to ensure that any data in your fragment is saved and restored as well.

How to Save fragment state in Android studio?

The first key is that we can save our Fragments' state ourselves using FragmentManager. saveInstanceState(Fragment) . Before removing the Fragment, call this method to get a Bundle containing your Fragment's saved state! The second key is restoring state.

What is saving state in Android?

The saveState() method allows your component to return a Bundle containing any state that should be saved from that component.

What are fragments in Android?

A Fragment represents a reusable portion of your app's UI. A fragment defines and manages its own layout, has its own lifecycle, and can handle its own input events. Fragments cannot live on their own--they must be hosted by an activity or another fragment.


1 Answers

I looked through the Android source code for the message in stack trace and found out that it comes from ActivityManagerService.handleAppDiedLocked() method. Descriptions of this method says:

"Main function for removing an existing process from the activity manager as a result of that process going away. Clears out all connections to the process."

This happens when application gets killed. It can be killed by the system, by another application (e.g. a task manager app) or when the application gets finished by itself (e.g. System.exit(0)).

like image 58
sergej shafarenka Avatar answered Sep 20 '22 04:09

sergej shafarenka