首先让我说这不是关于滚动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;
}
答案 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中数据的强大代表。
但我也有困惑。在这种情况下,您的方法和我的方法都假定内容更改意味着加载更多数据。但是,如果您确实需要重新加载所有数据呢?你将使用什么样的通知?