告诉Android AsyncTaskLoader以检索更多数据

时间:2013-06-03 14:35:24

标签: android android-loadermanager android-loader

首先让我说这不是关于滚动ListViews的问题。我不想知道如何判断用户何时滚动到列表的底部,所以请不要给我这个问题的答案,或者将其标记为重复。

我正在使用一个扩展AsyncTaskLoader的类来使用来自Web服务的数据填充ListView。

最初,我加载了50件商品,一切都很顺利。我需要知道如何告诉Loader以递增方式加载下50个项目。我理解在ListView代码中要执行此操作,但我无法找出告诉Loader我想加载更多数据的最佳方法而不重置它并再次加载所有内容。

再次,澄清一点,我在这里试图解决的问题只是通知加载器需要加载更多数据。它已经知道如何在第二次调用loadInBackground()时加载更多数据,并且ListView已经知道在何时/何时通知Loader,问题只是如何

一些相关代码:

@Override
public void onActivityCreated(Bundle savedInstanceState)
{
    super.onActivityCreated(savedInstanceState);

    m_adapter = new SearchAdapter(getActivity());
    setListAdapter(m_adapter);

    // if the loader doesn't already exist, one will be created
    // otherwise the existing loader is reused so we don't have
    // to worry about orientation and other configuration changes
    getLoaderManager().initLoader(SEARCH_LOADER_ID, null, this);
}

@Override
public Loader<List<Result>> onCreateLoader(int id, Bundle args)
{
    String query = args != null ? args.getString(QUERY_KEY) : "";
    return new SearchLoader(getActivity(), query);
}

private class SearchAdapter extends ArrayAdapter<Result>
{
    // ...

    @Override
    public View getView(int position, View convertView, ViewGroup parent)
    {

        // ...

        if (position == getCount() - 2)
            // TODO: Need to notify Loader here

        // ...
    }
}

private static class SearchLoader extends OurAsyncTaskLoader<List<Result>>
{
    public SearchLoader(Context context, String query)
    {
        super(context);

        m_query = query;
        m_data = Lists.newArrayList();
        m_loadedAllResults = false;
    }

    @Override
    public List<Result> loadInBackground()
    {
        if (m_loadedAllResults)
            return m_data;

        // the Loader implementation does a == check rather than a .equals() check
        // on the data, so we need this to be a new List so that it will know we have
        // new data
        m_data = Lists.newArrayList(m_data);

        MyWebService service = new MyWebService();
        List<Result> results = service.getResults(m_query, m_data.size(), COUNT);
        service.close();

        if (results == null)
            return null;

        if (results.size() < COUNT)
            m_loadedAllResults = true;

        for (Result result : results)
            m_data.add(result)

        return m_data;
    }

    private static final int COUNT = 50;

    private final String m_query;

    private boolean m_loadedAllResults;
    private List<Result> m_data;
}

2 个答案:

答案 0 :(得分:3)

我想出了一种有效的方法。在我的SearchAdapter#getView()方法中,我有以下代码:

private class SearchAdapter extends ArrayAdapter<Result>
{
    // ...

    @Override
    public View getView(int position, View convertView, ViewGroup parent)
    {

        // ...

        if (position == getCount() - 2)
            getLoaderManager().getLoader(SEARCH_LOADER_ID).onContentChanged();
        // ...
    }
}

我仍然想知道这是否是“最佳实践”方式,但它现在似乎解决了我的问题。

答案 1 :(得分:0)

在您的场景中,我建议您使用ForceLoadContentObserver,您可以使用URI绑定到ContentResolver。像这样:

class SearchLoader ....
     ForceLoadContentObserver contentObserver = new ForceLoadContentObserver();
     ....

     @Override
     public void onStartLoading() {
        if (cacheResult == null || takeContentChanged()) { // This will see if there's a change notification to observer and take it.
            onForceLoad();
        } else {
            deliverResult(cacheResult);
        }
     }

     @Override
     public Result loadInBackground() {
        Result result = loadResult();
        // notification uri built upon Result.BASE_URI so it receives all notifications to BASE_URI.
       getContext().getContentResolver().registerContentObserver(result.getNotificationUri(), true, contentObserver);
     }

     @Override
     public void onReset() {
        // ... Do your clean stuff...
        getContext().getContentResolver().unregisterContentObserver(contentObserver);
     }

     ...
}

因此,您可以使用以下方式在任何地方通知您的更改:

context.getContentResolver().notifyChanged(Result.BASE_URI, null);

即使持有加载程序的活动在后台或不能传递的结果。而且您不需要获取加载器的实例。

所有通知都是围绕Uri的对象。 Uri是Android中数据的强大代表。

但我也有困惑。在这种情况下,您的方法和我的方法都假定内容更改意味着加载更多数据。但是,如果您确实需要重新加载所有数据呢?你将使用什么样的通知?