Adapter在Android应用程序中起着非常重要的作用,应用也非常广泛,它可看作是数据源和UI组件之间的桥梁,其中Adapter、数据和UI之间的关系,可以用下图表示:

BaseAdapter就Android应用程序中经常用到的基础数据适配器,它的主要用途是将一组数据传到像ListView、Spinner、Gallery及GridView等UI显示组件,它是继承自接口类Adapter。

在ListView的使用中,有时候还需要在里面加入按钮等控件,实现单独的操作。也就是说,这个ListView不再只是展示数据,也不仅仅是这一行要来处理用户的操作,而是里面的控件要获得用户的焦点。读者可以试试用SimpleAdapter添加一个按钮到ListView的条目中,会发现可以添加,但是却无法获得焦点,点击操作被ListView的Item所覆盖。这时候最方便的方法就是使用灵活的适配器BaseAdapter了。

使用BaseAdapter必须写一个类继承它,同时BaseAdapter是一个抽象类,继承它必须实现它的方法。BaseAdapter的灵活性就在于它要重写很多方法,看一下有哪些方法,如图4-35所示为继承自BaseAdapter的SpeechListAdapter所实现的方法,其中最重要的即为getView()方法。这些方法都有什么作用呢?我们通过分析ListView的原理来为读者解答。

当系统开始绘制ListView的时候,首先调用getCount()方法。得到它的返回值,即ListView的长度。然后系统调用getView()方法,根据这个长度逐一绘制ListView的每一行。也就是说,如果让getCount()返回1,那么只显示一行。而getItem()和getItemId()则在需要处理和取得Adapter中的数据时调用。那么getView如何使用呢?如果有10000行数据,就绘制10000次?这肯定会极大的消耗资源,导致ListView滑动非常的慢,那应该怎么做呢?通过一个例子来讲解如何在使用BaseAdapter的时候优化ListView的显示。例子中将上一节中的ImageView换成Button,并且处理Button的点击事件,其中对ListView的显示做了优化。

publicclassMyListViewBaseextendsActivity{privateListViewlv;//定义一个动态数组ArrayList<HashMap<String,Object>>listItem;/**Calledwhentheactivityisfirstcreated.*/@OverridepublicvoidonCreate(BundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.main);lv=(ListView)findViewById(R.id.lv);MyAdaptermAdapter=newMyAdapter(this);//得到一个MyAdapter对象lv.setAdapter(mAdapter);//为ListView绑定Adapter/**为ListView添加点击事件*/lv.setOnItemClickListener(newOnItemClickListener(){@OverridepublicvoidonItemClick(AdapterView<?>arg0,Viewarg1,intarg2,longarg3){Log.v("MyListViewBase","你点击了ListView条目"+arg2);//在LogCat中输出信息}});}/**添加一个得到数据的方法,方便使用*/privateArrayList<HashMap<String,Object>>getDate(){ArrayList<HashMap<String,Object>>listItem=newArrayList<HashMap<String,Object>>();/**为动态数组添加数据*/for(inti=0;i<30;i++){HashMap<String,Object>map=newHashMap<String,Object>();map.put("ItemTitle","第"+i+"行");map.put("ItemText","这是第"+i+"行");listItem.add(map);}returnlistItem;}/**新建一个类继承BaseAdapter,实现视图与数据的绑定*/privateclassMyAdapterextendsBaseAdapter{privateLayoutInflatermInflater;//得到一个LayoutInfalter对象用来导入布局/**构造函数*/publicMyAdapter(Contextcontext){this.mInflater=LayoutInflater.from(context);}@OverridepublicintgetCount(){returngetDate().size();//返回数组的长度}@OverridepublicObjectgetItem(intposition){returnnull;}@OverridepubliclonggetItemId(intposition){return0;}/**书中详细解释该方法*/@OverridepublicViewgetView(finalintposition,ViewconvertView,ViewGroupparent){ViewHolderholder;//观察convertView随ListView滚动情况Log.v("MyListViewBase","getView"+position+""+convertView);if(convertView==null){convertView=mInflater.inflate(R.layout.item,null);holder=newViewHolder();/**得到各个控件的对象*/holder.title=(TextView)convertView.findViewById(R.id.ItemTitle);holder.text=(TextView)convertView.findViewById(R.id.ItemText);holder.bt=(Button)convertView.findViewById(R.id.ItemButton);convertView.setTag(holder);//绑定ViewHolder对象}else{holder=(ViewHolder)convertView.getTag();//取出ViewHolder对象}/**设置TextView显示的内容,即我们存放在动态数组中的数据*/holder.title.setText(getDate().get(position).get("ItemTitle").toString());holder.text.setText(getDate().get(position).get("ItemText").toString());/**为Button添加点击事件*/holder.bt.setOnClickListener(newOnClickListener(){@OverridepublicvoidonClick(Viewv){Log.v("MyListViewBase","你点击了按钮"+position);//打印Button的点击信息}});returnconvertView;}}/**存放控件*/publicfinalclassViewHolder{publicTextViewtitle;publicTextViewtext;publicButtonbt;}}

运行效果如图4-36所示。

4-36使用BaseAdapterListVie

还需要注意的是,Button会抢夺ListView的焦点,需要将Button设置为没有焦点。设置非常简单,只需要在xmlButton标签下加入一行:android:focusable=“false”代码就可以了。在LogCat观察点击后输出的信息,如图4-37所示。


4-37点击ListView条目和Button得到的输出

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

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

4-38滚动ListView输出的convertView的值

总结一下,这节介绍了用BaseAdapter来绑定ListView的数据。因为BaseAdapter非常灵活,使用也相对较其他控件麻烦。同时ListView的优化问题也值得读者去研究,一个流畅的ListView会带来更好的用户体验。


更多相关文章

  1. Android系列之GreenDao数据升级和加密(三)
  2. Android(安卓)2.3 input输入事件处理
  3. 在Android上实现树形控件
  4. Android(安卓)Virtualview:淘宝、天猫 又一个动态化、高性能的UI
  5. Android(安卓)访问Android(安卓)Wear数据层Api——同步Data Item
  6. Androidの解决自动旋转导致activity重启问题
  7. Android开发系列之调用WebService
  8. Android:快速修改ramdisk.img脚本
  9. Android(安卓)Studio使用webservice远程访问数据库SQL Server 20

随机推荐

  1. Android应用开发提高系列(1)——《Practica
  2. Android(安卓)AppWidget(桌面小部件-音乐
  3. Android: Android(安卓)设计思想
  4. ClockworkMod Recovery瀹炵幇搴旂敤andro
  5. 谈话....
  6. Android 的输入法适配设置windowSoftInpu
  7. Android(安卓)Q Beta登场,新特性抢先看!
  8. Android时代的赢创之路
  9. Android UI开发第二十四篇——Action Bar
  10. Android 中,应用程序需要的图片资源如何针