Reputation: 586
I noticed strange situation using Android Loaders and Fragments. When I invoke LoaderManager.initLoader() after orientation change onLoadFinished is not called (although documentation suggests I should be prepared for this) but it is called twice after this. Here is link to post in google groups which describe the same situation https://groups.google.com/forum/?fromgroups#!topic/android-developers/aA2vHYxSskU . I wrote sample application in which I only init simple Loader in Fragment.onActivityCreated() to check if this happens and it does. Anyone noticed this?
Upvotes: 56
Views: 20514
Reputation: 745
If implementing AppCompatActivity, double check you are using getSupportLoaderManager() in all cases (destroyLoader/initLoader etc). I had mistakenly used getSupportLoaderManager() in conjunction with a getLoaderManager() and suffered the same issue.
Upvotes: 0
Reputation: 94
initLoader documentation says,
If at the point of call the caller is in its started state, and the requested loader already exists and has generated its data, then callback onLoadFinished(Loader, D)
I suggest you to implement something like onStartLoading function at this sample
For quick test you can try:
@Override protected void onStartLoading() {
forceLoad();
}
This launch loadInBackground function and then onLoadFinished in Fragment.
Any way, if you attach some code i'll try to give you more help.
Upvotes: 6
Reputation: 1413
i have face this issue.but i while used to call the destroyloader(YOUR_ID)
in loaderfinished methods. then the loader not again call the backgrdound task twice.
Upvotes: 0
Reputation: 326
I solved the problem of onLoadFinished being called twice like this. In your Fragment.onActivityCreated() init your Loader like this
if (getLoaderManager().getLoader(LOADER_ID) == null) {
getLoaderManager().initLoader(LOADER_ID, bundle, loaderCallbacks);
} else {
getLoaderManager().restartLoader(LOADER_ID, bundle, loaderCallbacks);
}
here loaderCallbacks implements your usual Loader callbacks
private LoaderManager.LoaderCallbacks<T> loaderCallbacks
= new LoaderManager.LoaderCallbacks<T>() {
@Override
public Loader<T> onCreateLoader(int id, Bundle args) {
...
...
}
@Override
public void onLoadFinished(Loader<T> loader, T data) {
...
...
}
@Override
public void onLoaderReset(Loader<T> loader) {
...
...
}
};
Upvotes: 5
Reputation: 3441
Since all searching for this subject inevitably ends up here, I just wanted to add my experience. As @jperera said, the culprit was that LoaderManager will call onLoadFinished() if the loaders already exist. In my case, I had fragments in a FragmentPager and scrolling 2 tabs away and then scrolling next to it again would cause my old fragment to begin creating itself.
Since placing initLoader() inside onCreate() also causes double callbacks, I placed the initLoader() inside onResume(). But the sequence of events ends up being onCreate(), LoaderManager calls callbacks since loaders exist, then onResume() is called, triggering another initLoader() and onLoadFinished() sequence. IE, another double callback.
I found a quick solution by "Matt". After all your data is loaded (if you have more than one loader), destroy all of the loaders so their callbacks won't be called an extra time.
Upvotes: 0
Reputation: 59
You can also compare the data object in onLoadFinished(Loader loader, Object data). If the data object matches one you already have, you can just not do anything when onLoadFinished is called. For example:
public void onLoadFinished(Loader loader, Object data) {
if(data != null && mData != data){
//Do something
}
}
Upvotes: 0
Reputation: 9450
When calling initLoader
from onActivityCreated
you can detect rotation :
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
if (savedInstanceState == null) {
// fresh new fragment, not orientation/config change
getLoaderManager().initLoader(YOUR_LOADER_ID, null, mCallbacks);
}
...
}
This way the loader behaves as expected resulting in single onLoadFinished
call.
It is not called on rotation anymore, so if you want loader's data, you can keep it in your fragment, for instance by overriding onSaveInstanceState
.
Edit:
I just realized that onLoadFinished
won't be called if rotation happens during loader's loadInBackground
. To fix this you'd still need to call initLoader
after rotation if the data from loader not yet available.
Hope that helps.
Upvotes: 2
Reputation: 1292
The problem is that it called twice:
1. from Fragment.onStart
2. from FragmentActivity.onStart
The only difference is that in Fragment.onStart it checks if mLoaderManager != null. What this means is if you call getLoadManager before onStart, like in onActivityCreated, it will get/create load manager and it will be called. To avoid this you need to call it later, like in onResume.
Upvotes: 3
Reputation: 3847
This problem manifested itself for me with a CursorLoader returning a Cursor that was already closed:
android.database.StaleDataException: Attempted to access a cursor after it has been closed.
I'd guess this is a bug or an oversight. While moving initLoader() into onResume may work, what I was able to do was remove the Loader when I'm done with it:
To start the loader (in my onCreate):
getLoaderManager().initLoader(MUSIC_LOADER_ID, null, this);
Then after I'm done with it (basically at the end of onLoadFinished)
getLoaderManager().destroyLoader(MUSIC_LOADER_ID);
This seems to behave as expected, no extra calls.
Upvotes: 32
Reputation: 6451
You can put the initLoader() method inside your Fragment's onResume() callback; then the Loader's onLoadFinished() will not be called twice anymore.
@Override
public void onResume()
{
super.onResume();
getLoaderManager().initLoader(0, null, this);
}
Upvotes: 39