ListView的深入学习

ListView通常有两个职责:

将数据填充到布局 ; 处理用户的点击选择操作

二。创建ListView需要3个元素

  ListView的每一列的View

  View的数据或者图片

  连接数据与ListView的Adapter

三。常用的适配器

 

 

Adapter

含义

ArrayAdapter<T>

用来绑定一个数组,支持泛型操作

SimpleAdapter

用来绑定在xml中定义的控件对应的数据

SimpleCursorAdapter

用来绑定游标得到的数据

BaseAdapter

通用的基础适配器

 

四,ArrayAdapter

代码:

String[] str=new String[]{"1","2","3"};

lv.setAdapter(new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,str) );

 

五。SimpleAdapter

代码:

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("ItemImage",R.drwaable.icon);

      map.put("ItemTitle","第"+i+“行”);

      map.put("ItemText",i+"行");

      listItem,add(map);    

    }

//this,存放的数据,每一行的布局,动态数组中的数据源的键对应到布局view中

SimpleAdapter  smadapter=new SimpleAdapter(this,listItem,R.layout.item,

                  new String[]{" ItemImage","ItemTitle","ItemText"}

                  new int[]{R.id.ItemImage,R.id.ItemTitle,R.id.ItemText});

lv.setAdapter(smadapter);

 

主意:使用SimpleAdapter一般都是用HashMap构成的列表,列表的每一节对应LisView的每一行。通过SimpleAdapter的构造函数,将HashMap的每个键的数据映射到布局文件的控件上

 

六。BaseAdapter

在ListView的使用中,还需要加入按钮等空间,实现单独的操作,用SimpleAdapter无法获取焦点,点击操作会被ListView的item覆盖,所以用到来Baseadapter。

使用BaseAdapter必须写一个类来继承它,以为它是一个抽象类,重写4个方法,最重要的方法是getView();

当开始绘制ListView的时候,首先调用getCount()方法,得到它的返回值,即listView的长度;

然后调用getView()方法,根据长度逐一绘制ListView的每一行;

然后getItem()和getItemId()在需要处理和取得Adapter中的数据时调用

 

代码:

//创建MyApdater继承BaseAdapter

MyAdapter mAdapter=new MyAdapter(this);

lv.setAdapter(mAdapter);

lv.setOnItemClickListener(new OnItemClickListener(){

  public void onItemClick(AdapterView<?>arg0,View arg1,int arg2,long arg3){

    log.v("你点击了"+arg2);

    }

});

//方法:得到数据

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

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

  //为动态数组添加数据

  for(int i=0,i<10,i++){

      HashMap<String ,Object> map=new HashMap<String,Object>();

      map.put("ItemTitle","第"+i+"行");

      map.pt("ItemText","第'+i+"行");

      listItem.add(map);

        }

    return listItem;

}

 

//BaseAdapter的实体类MyAdapter

private class MyAdapter extends BaseAdpater{

  //得到一个LayoutInfalter对象来导入布局

  private LayoutInflater mInflater;

  public MyAdapter(Context context){

  this.mInflater=LayoutInflater.from(context);

  }

  // 重写的方法

  public int getCount(){

  return getDate().size();  //返回数组的长度

  }

  public Object getItem(int position){

  return null;

  }

  public long getItemId(int position){

  return 0;

  }

  //重写最重要的方法

  pubic View getView(final int position,View convertView,ViewGroup parent){

    ViewHolder holder;

    //观察convertView随listVIew的滚动情况

    Log.v("my","getView"+position+"---"+convertView);

    if(convertView==null){

        convertView=nInflater.inflate(R.layout.item,null);

        holder=new ViewHolder();

        // 实力化控件

        holder.title=(TextView)convertVIew.finViewByid(R.id.xxx);

        holder.text=(TextView)convertVIew.finViewByid(R.id.xxx);

        holder.tbutton=(Button)convertVIew.finViewByid(R.id.xxx); 

        convertView,setTag(holder);     //绑定ViewHolder对象

      }else{

        //取出ViewHolder对象,ViewHolder在下面有定义

        holder=(ViewHolder)contretView.getTag();

        //设置TextVIew显示的内容,即我们存放在动态数组中的数据

        holder.title.setText(getDate().get(position).get("ItemTitle").toString());
            holder.text.setText(getDate().get(position).get("ItemText").toString());
            
            /*为Button添加点击事件*/              holder.bt.setOnClickListener(new OnClickListener() {
                
                @Override
                publicvoid onClick(View v) {
                Log.v("MyListViewBase", "你点击了按钮" + position);                                //打印          Button的点击信息                    
                }
            });

        return convertView;

          }

  }

  //存放数据的ViewHolder

   public final class ViewHolder{

          public TextView title;      

           public TextView text; 

           public Button bt;

    }

 

}

例子

主意:

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

从图4-38中可以看出,当启动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对控件的层层查询,而是快速定位到控件。

 

posted @ 2015-10-31 13:40  冷月舞痕  阅读(567)  评论(0编辑  收藏  举报