我遇到了一些我从书中改编的BaseAdapter代码的问题。我一直在我的应用程序中使用此代码的变体,但只是在滚动一个长列表时才意识到ListView中的项目变得混乱而且并不显示所有元素。
描述确切的行为非常困难,但很容易看出你是否采用了50个项目的排序列表并开始上下滚动。
class ContactAdapter extends BaseAdapter {
ArrayList<Contact> mContacts;
public ContactAdapter(ArrayList<Contact> contacts) {
mContacts = contacts;
}
@Override
public int getCount() {
return mContacts.size();
}
@Override
public Object getItem(int position) {
return mContacts.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View view;
if(convertView == null){
LayoutInflater li = getLayoutInflater();
view = li.inflate(R.layout.groups_item, null);
TextView label = (TextView)view.findViewById(R.id.groups_item_title);
label.setText(mContacts.get(position).getName());
label = (TextView)view.findViewById(R.id.groups_item_subtitle);
label.setText(mContacts.get(position).getNumber());
}
else
{
view = convertView;
}
return view;
}
}
答案 0 :(得分:13)
您只是在首次创建数据时将数据放入TextView
小部件中。你需要移动这四行:
TextView label = (TextView)view.findViewById(R.id.groups_item_title);
label.setText(mContacts.get(position).getName());
label = (TextView)view.findViewById(R.id.groups_item_subtitle);
label.setText(mContacts.get(position).getNumber());
在if
/ else
块之后和方法返回之前,因此您更新TextView
小部件,无论您是回收行还是创建新行。
答案 1 :(得分:5)
为了进一步澄清CommonsWare的答案,这里有更多信息:
li.inflate 操作(这里需要从XML解析行的布局并创建适当的View对象)由 if(convertView == null)<包装/ em>效率声明,因此每次弹出视图时,同一物体的膨胀不会一次又一次地发生。
但是,getView方法的其他部分用于设置其他参数,因此不应包含在 if(convertView == null){} ... else {} 语句中。
在许多常见的此方法的实现,一些TextView的标签,或ImageView的元素的ImageButton需要由值从列表[位置]填充,使用 findViewById 和之后的 .setText < / em>或 .setImageBitmap 操作。 这些操作必须在两者之后通过通货膨胀创建一个视图,并且获取现有视图(如果不是null)(例如刷新时)。
此解决方案应用于ListView ArrayAdapter的另一个好例子显示在https://stackoverflow.com/a/3874639/978329