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

来源:互联网 发布:淘宝图片轮播什么意识 编辑:程序博客网 时间:2024/06/05 16:01

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对控件的层层查询,而是快速定位到控件。

0 0
原创粉丝点击