android ListView 的无限适配器
Endless adapter for android ListView
我使用 Ion 库从服务器接收数据。我有问题,我的代码多次发出请求,所以我的列表中有数据重复。
我的适配器代码:
public class ArticleAdapter extends ArrayAdapter<Article> {
Future<List<Article>> loadingOfPrevious;
static class ArticleHolder {
ImageView articleImage;
TextView articleCaption;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ArticleHolder holder;
if (convertView == null) {
convertView = layoutInflater.inflate(R.layout.list_view_item, null);
holder = new ArticleHolder();
holder.articleImage = (ImageView) convertView.findViewById(R.id.articleImage);
holder.articleCaption = (TextView) convertView.findViewById(R.id.articleCaption);
convertView.setTag(holder);
} else {
holder = (ArticleHolder) convertView.getTag();
}
Article article = getItem(position);
holder.articleCaption.setText(article.getTitle());
Ion.with(holder.articleImage)
.placeholder(R.drawable.placeholder)
.error(R.drawable.default_article_image)
//.crossfade(true)
.load(article.getImageUrl());
// we're near the end of the list adapter, so load more items
if (position >= getCount() - 3) {
loadPrevious(getItem(getCount() - 1).getId());
}
return convertView;
}
private void loadPrevious(long id) {
// don't attempt to load more if a load is already in progress
if (loadingOfPrevious != null && !loadingOfPrevious.isDone() && !loadingOfPrevious.isCancelled()) {
return;
}
String url = "http://example.com/rest-api/"
url = url + "?id=" + id;
url = url + "&count=" + 30;
// This request loads a URL as JsonArray and invokes
// a callback on completion.
final String articleUrl = url;
loadingOfLatest = Ion.with(getContext())
.load(articleUrl)
.as(new TypeToken<List<Article>>() {
})
.setCallback(new FutureCallback<List<Article>>() {
@Override
public void onCompleted(Exception e, List<Article> result) {
// this is called back onto the ui thread, no Activity.runOnUiThread or Handler.post necessary.
if (e != null) {
Toast.makeText(getContext(), "Error.", Toast.LENGTH_LONG).show();
return;
}
// add the article
if (result != null) {
Collections.reverse(result);
for (int i = 0; i < result.size(); i++) {
add(result.get(i));
}
notifyDataSetChanged();
}
}
});
}
}
我认为问题出在这部分:
// we're near the end of the list adapter, so load more items
if (position >= getCount() - 3) {
loadPrevious(getItem(getCount() - 1).getId());
}
我这样做的方法是在 ListView 上使用 OnScrollListener。当用户滚动并且 firstVisibleItem + visibleItemCount
接近列表限制时,开始加载服务器数据。
数据加载完成后,使用新数据更新您的列表适配器并调用 onNotifyDataSetChanged()
。然后该列表将重新显示您的更新数据。
你说得对!就是因为这段代码。
if (position >= getCount() - 3) {
loadPrevious(getItem(getCount() - 1).getId());
}
根据你的情况,"loadPrevious()"会调用两次。
"getView()" 每次 return 一个视图。因此,对于最后两项,您的条件为真,然后调用 "loadPrevious()" 两次。
所以改变并尝试这样
if (position >= getCount() - 2) {
loadPrevious(getItem(getCount() - 1).getId());
}
或
重新检查此代码
if (loadingOfPrevious != null && !loadingOfPrevious.isDone() && !loadingOfPrevious.isCancelled()) {
return;
}
顺便说一句,我想建议尝试其他方法,比如使用库,自定义列表视图
Android Endless List
我使用 Ion 库从服务器接收数据。我有问题,我的代码多次发出请求,所以我的列表中有数据重复。 我的适配器代码:
public class ArticleAdapter extends ArrayAdapter<Article> {
Future<List<Article>> loadingOfPrevious;
static class ArticleHolder {
ImageView articleImage;
TextView articleCaption;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ArticleHolder holder;
if (convertView == null) {
convertView = layoutInflater.inflate(R.layout.list_view_item, null);
holder = new ArticleHolder();
holder.articleImage = (ImageView) convertView.findViewById(R.id.articleImage);
holder.articleCaption = (TextView) convertView.findViewById(R.id.articleCaption);
convertView.setTag(holder);
} else {
holder = (ArticleHolder) convertView.getTag();
}
Article article = getItem(position);
holder.articleCaption.setText(article.getTitle());
Ion.with(holder.articleImage)
.placeholder(R.drawable.placeholder)
.error(R.drawable.default_article_image)
//.crossfade(true)
.load(article.getImageUrl());
// we're near the end of the list adapter, so load more items
if (position >= getCount() - 3) {
loadPrevious(getItem(getCount() - 1).getId());
}
return convertView;
}
private void loadPrevious(long id) {
// don't attempt to load more if a load is already in progress
if (loadingOfPrevious != null && !loadingOfPrevious.isDone() && !loadingOfPrevious.isCancelled()) {
return;
}
String url = "http://example.com/rest-api/"
url = url + "?id=" + id;
url = url + "&count=" + 30;
// This request loads a URL as JsonArray and invokes
// a callback on completion.
final String articleUrl = url;
loadingOfLatest = Ion.with(getContext())
.load(articleUrl)
.as(new TypeToken<List<Article>>() {
})
.setCallback(new FutureCallback<List<Article>>() {
@Override
public void onCompleted(Exception e, List<Article> result) {
// this is called back onto the ui thread, no Activity.runOnUiThread or Handler.post necessary.
if (e != null) {
Toast.makeText(getContext(), "Error.", Toast.LENGTH_LONG).show();
return;
}
// add the article
if (result != null) {
Collections.reverse(result);
for (int i = 0; i < result.size(); i++) {
add(result.get(i));
}
notifyDataSetChanged();
}
}
});
}
}
我认为问题出在这部分:
// we're near the end of the list adapter, so load more items
if (position >= getCount() - 3) {
loadPrevious(getItem(getCount() - 1).getId());
}
我这样做的方法是在 ListView 上使用 OnScrollListener。当用户滚动并且 firstVisibleItem + visibleItemCount
接近列表限制时,开始加载服务器数据。
数据加载完成后,使用新数据更新您的列表适配器并调用 onNotifyDataSetChanged()
。然后该列表将重新显示您的更新数据。
你说得对!就是因为这段代码。
if (position >= getCount() - 3) {
loadPrevious(getItem(getCount() - 1).getId());
}
根据你的情况,"loadPrevious()"会调用两次。
"getView()" 每次 return 一个视图。因此,对于最后两项,您的条件为真,然后调用 "loadPrevious()" 两次。 所以改变并尝试这样
if (position >= getCount() - 2) {
loadPrevious(getItem(getCount() - 1).getId());
}
或
重新检查此代码
if (loadingOfPrevious != null && !loadingOfPrevious.isDone() && !loadingOfPrevious.isCancelled()) {
return;
}
顺便说一句,我想建议尝试其他方法,比如使用库,自定义列表视图
Android Endless List