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

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值