您的位置:首页 > 其它

ListView最基本的使用方法BaseAdapter(三)

2016-11-16 14:34 453 查看
ListView最基本的使用方法BaseAdapter(三)

在ListView的用法(二)中,我在每个Item的布局中加入了一个Button,但是没法给这个Button添加响应事件,因为它无法获得焦点,点击操作会被ListView的Item点击事件覆盖,这时候我们就得使用更加灵活的BaseAdapter了。也就是说,这个ListView不再只是展示数据,也不仅仅是这一行要来处理用户的操作,而是里面的控件要获得用户的焦点。

先来个最简单的代码列子,其中listlayout.xml和listitem.xml没变,和上一篇文章中的一模一样。

首先是自己写的PersonalAdapter继承了BaseAdapter,BaseAdapter是一个抽象类,需要覆盖几个方法;

public class PersonalAdapter extends BaseAdapter {

private LayoutInflater mInflater;
private Context context;

public class ViewHolder{
public ImageView icon;
public TextView name;
public Button button;
}

public PersonalAdapter(Context context){
this.context = context;
this.mInflater = LayoutInflater.from(context);
}

@Override
public int getCount(){
return getData().size();
}

@Override
public long getItemId(int arg0){
return 0;
}

@Override
public Object getItem(int arg0){
return  null;
}

private ArrayList<HashMap<String,Object>> getData(){

ArrayList<HashMap<String,Object>> listItem = new ArrayList<HashMap<String, Object>>();

for(int i=0;i<10;i++){
HashMap<String,Object> map = new HashMap<String,Object>();
map.put("icon",R.drawable.pic1);
map.put("name","小明"+i+"号");
map.put("button","第"+i+"列");
listItem.add(map);
}

return listItem;
}

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

ViewHolder holder = null;

if(convertView == null){   //如果缓存中已有,不用重新加载,提高性能
holder = new ViewHolder();
convertView = mInflater.inflate(R.layout.listitem,null);
holder.icon = (ImageView)convertView.findViewById(R.id.icon);
holder.name = (TextView)convertView.findViewById(R.id.name);
holder.button = (Button)convertView.findViewById(R.id.button);
convertView.setTag(holder);
}else{
holder = (ViewHolder)convertView.getTag();
}

holder.icon.setImageResource((Integer)getData().get(position).get("icon"));
holder.name.setText(getData().get(position).get("name").toString());
holder.button.setText(getData().get(position).get("button").toString());

holder.button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
new AlertDialog.Builder(context)
.setIcon(R.drawable.pic2)
.setTitle("谢谢你点击我"+position)
.setPositiveButton("确定",
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog,
int which) {
// TODO Auto-generated method stub

}
}).setNegativeButton("取消", null).create()
.show();
}
});

return convertView;
}
}


MainActivity的代码,非常简单,整个过程还是很清晰的。

public class MainActivity extends Activity {

private ListView listView;              //list布局
private PersonalAdapter adapter;  //链接list控件和数据的adapter

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.listlayout);

listView = (ListView)findViewById(R.id.myList);

adapter = new PersonalAdapter(this);
listView.setAdapter(adapter);
}
}


代码中getView()方法不容易理解。其实完全可以不用所谓的convertView和ViewHolder,直接导入布局并且设置控件显示的内容就可以了。但是这意味着有多少行数据就需要绘制多少行ListView,这显然是不可取的。

这里采用了一种优化的方法。当启动Activity呈现第一屏ListView的时候,convertView为零。当用户向下滚动ListView时,上面的条目变为不可见,下面出现新的条目。这时候convertView不再为空,而是创建了一系列的convertView的值。当又往下滚一屏的时候,发现第11行的容器用来容纳第22行,第12行的容器用来容纳第23行。也就是说convertView相当于一个缓存,开始为0,当有条目变为不可见,它缓存了它的数据,后面再出来的条目只需要更新数据就可以了,这样大大节省了系统资料的开销。

还可以继续优化。虽然重复利用了已经绘制的view,但是要得到其中的控件,需要在控件的容器中通过findViewById的方法来获得。如果这个容器非常复杂,这显然会增加系统资源的开销。在上面的例子中,引入了Tag的概念。或许不是最好的办法,但是它确实能使ListView变得更流畅。代码中,当convertView为空时,用setTag()方法为每个View绑定一个存放控件的ViewHolder对象。当convertView不为空,重复利用已经创建的view的时候,使用getTag()方法获取绑定的ViewHolder对象,这样就避免了findViewById对控件的层层查询,而是快速定位到控件。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息