BaseAdapter中重写getview的心得以及发现convertView回收的机制

来源:互联网 发布:网络挖矿是啥意思 编辑:程序博客网 时间:2024/05/19 02:25

以前一直在用BaseAdapter,对于其中的getview方法的重写一直不太清楚。今天终于得以有空来探究它的详细机制。
下面先讲讲我遇到的几个问题:
一.View getview(int position, View convertview, ViewGroup parent )中的第二个参数是什么含义;
二.View的SetTag和getTag方法的用途;
先来解决第一个问题:
android SDK中这样讲参数 convertview :
the old view to reuse, if possible. 
Note: You should check that this view is non-null and of an appropriate type before using.

 If it is not possible to convert this view to display the correct data, this method can create a new view.
翻译:
如果可以的话,这是旧View(这里不便翻译有的人翻成视图)的重用。 建议:在用之前,你应该检查这个View是
不是非空,是不是一个合适的类型。
如果不可能让这个VIew去显示一个恰当的数据,这个方法会创建一个新的View。

如果我们要做的是一个ListView,在手机上显示的只有那么几条Item,而整个ListView可能有很长,可能是100条
甚至是上万条,总不能让这么多条Item都驻留在内存中,所以android为你准备了一套机制,就是Recycler(反复循
环器),他的具体工作原理可以到 http://www.cnblogs.com/xiaowenji/archive/2010/12/08/1900579.html去看。
但是有些地方他没有讲清,所以我再讲一下。先把代码贴出来
布局文件main.xml

<?xml version="1.0" encoding="utf-8"?>  <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"      android:layout_width="fill_parent"      android:layout_height="fill_parent"      android:orientation="vertical" >      <ListView          android:id="@+id/result"          android:layout_width="fill_parent"          android:layout_height="fill_parent"          android:cacheColorHint="#00000000" >  </ListView>  </LinearLayout> 

此处注意ListView的android:layout_height属性值为"fill_paternt",如果为“wrap_content"将会是另一种情况

adapter的代码ListViewAdapter.java

public class <strong>ListViewAdapter </strong>extends BaseAdapter {private List<PackageInfo> packageInfos;private Context context;private static View view;public AppManagerAdapter(Context context, List<PackageInfo> packageInfos) {this.packageInfos = packageInfos;this.context = context;}@Overridepublic int getCount() {// TODO Auto-generated method stubreturn packageInfos.size();}@Overridepublic Object getItem(int position) {// TODO Auto-generated method stubreturn getItem(position);}@Overridepublic long getItemId(int position) {// TODO Auto-generated method stubreturn position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {// Holder holder = null;if (convertView != null) {view = convertView;} else {convertView = View.inflate(context, R.layout.app_manager_item, null);}/* * TextView的修改部分必须放在if……else……后面,因为layoutInflater.inflate(R.layout. * app_manager_item, null); 在每次获取的是空的,不会获取到上回holder中textView的内容 */PackageInfo packageInfo = packageInfos.get(position);Drawable drawable = packageInfo.applicationInfo.loadIcon(context.getPackageManager());drawable.setBounds(0, 0, 80, 80);// int ic = packageInfo.applicationInfo.icon;// System.out.println(icon + " ||" + packageInfo.packageName);// 此方法无法获取到系统图标只能获取到本应用图标// textView.setCompoundDrawablesWithIntrinsicBounds(ic, 0, 0, 0);TextView textView = (TextView) convertView.findViewById(R.id.tv_appitem);textView.setText(packageInfo.applicationInfo.loadLabel(context.getPackageManager()));textView.setCompoundDrawables(drawable, null,null, null);return convertView;}}


运行程序之后发现屏幕上显示出的Item的convertview都为空,向下滑动新产生的Item的convetview都不为空。到此为止和上面链接中讲的是一致的,但是如果设置ListView的android:layout_height属性值为“wrap_content

之后,发现只有第一个Item的convertview为null其他的不为空。

虽然两种设置不同,结果也不同,但是convertview的机制没有变。

其实到此为止我们可以总结出convertview的机制了,就是在初始显示的时候,每次显示一个item都调用一次getview方法但是每次调用的时候covertview为空(因为还没有旧的view),当显示完了之后。如果屏幕移动了之后,并且导致有些Item(也可以说是view)跑到屏幕外面,此时如果还有新的item需要产生,则这些item显示时调用的getview方法中的convertview参数就不是null,而是那些移出屏幕的view(旧view),我们所要做的就是将需要显示的item填充到这些回收的view(旧view)中去,最后注意convertview为null的不仅仅是初始显示的那些item,还有一些是已经开始移入屏幕但是还没有view被回收的那些item。

最终我们用亲手写的代码实现了Recycler(反复循环器).


第二个问题其实应该在第一个问题中嵌套来讲,但是为了思路清晰我分开了:

view的setTag和getTag方法其实很简单,在实际编写代码的时候一个view不仅仅是为了显示一些字符串、图片,有时我们还需要他们携带一些其他的数据以便我们对该view的识别或者其他操作。于是android 的设计者们就创造了setTag(Object)方法来存放一些数据和view绑定,我们可以理解为这个是view 的标签也可以理解为view 作为一个容器存放了一些数据。而这些数据我们也可以通过getTag() 方法来取出来。

到这里setTag和getTag大家应该已经明白了。再回到上面的话题,我们通过convertview的setTag方法和getTag方法来将我们要显示的数据来绑定在convertview上。如果convertview 是第一次展示我们就创建新的Holder对象与之绑定,并在最后通过return convertview 返回,去显示;如果convertview 是回收来的那么我们就不必创建新的holder对象,只需要把原来的绑定的holder取出加上新的数据就行了。


0 0
原创粉丝点击