3

我有AsyncTaskLoader<List<String>>一个成员变量List<String>mAddresses。在loadInBackground,如果我创建一个新对象,填充它,然后返回它;onLoadFinished按预期调用,所以我可以更新适配器,从而更新视图。

    public List<String> loadInBackground()
    {
        // ListView updates properly only if I do this (strange?)
        mAddresses = new ArrayList<String>();

        // ... fill mAddresses ...

        return mAddresses;
    }

但是,如果我尝试使用相同的变量,即使它的值发生变化也会onLoadFinished被调用:

    public List<String> loadInBackground()
    {
        // ListView does not update with this!
        if(mAddresses == null)
        {
           mAddresses = new ArrayList<String>();
        }

        // ... fill mAddresses ...

        return mAddresses;
    }

任何人都可以对此发表评论吗?我是否应该创建一个新对象以每次都像这样返回?是否可以设置任何标志“对象已更改,因此即使它是同一个对象,请确保执行所有更新?

4

2 回答 2

6

如果数据已经加载(就像在加载器中的“mAddresses”变量中显示的那样),您需要调用 forceLoad() 以使其再次加载:

http://developer.android.com/reference/android/content/Loader.html#forceLoad%28%29



编辑: ......经过一些调查......回答你原来的问题,无济于事:'因为它没有'!

来自 LoaderManager 的源代码:

        // Notify of the new data so the app can switch out the old data before
        // we try to destroy it.
        if (mData != data || !mHaveData) {
            mData = data;
            mHaveData = true;
            if (mStarted) {
                callOnLoadFinished(loader, data);
            }
        }

http://grepcode.com/file/repository.grepcode.com/java/ext/com.google.android/android/4.3_r2.1/android/app/LoaderManager.java#LoaderManagerImpl.LoaderInfo.onLoadComplete%28android.content .Loader%2Cjava.lang.Object%29

重新解决您的问题,我假设您在 loadInBackground() 中不断添加到列表中,而不是清除并重新开始。最干净的方法是每次通过将现有列表作为参数传递给 ArrayList() 构造函数来构建一个新的 ArrayList,它将复制内容。
或者你可以重写deliverResult(),它是从主线程调用的,所以修改mAdapter是可以的。
或者忘记加载器,只使用更简单的 AsyncTask ......尽管这取决于您的数据实际来自何处。

于 2013-09-24T13:36:58.427 回答
1

尝试覆盖 onStartLoading 并调用 forceLoad()

@Override
protected void onStartLoading() {
    super.onStartLoading();
    forceLoad();
}

@Override
protected void onStopLoading() {
    super.onStopLoading();
    cancelLoad();
}
于 2016-08-31T14:43:02.397 回答