ListView BaseAdapter 简单分析

来源:互联网 发布:golang json map 遍历 编辑:程序博客网 时间:2024/06/05 11:47

接触listview 有一段时间了,一直没有好好分析,今天仔细分析了下流程,记录在此,也给新进的同学一个参考。

先附代码:

RecentAdapter:

package com.test.baseadapter;import java.util.ArrayList;import android.content.Context;import android.util.Log;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.widget.BaseAdapter;import android.widget.ImageView;import android.widget.TextView;public class RecentAdapter extends BaseAdapter {private ArrayList<String> mArrayList;private LayoutInflater mLayoutInflater;public RecentAdapter(Context context, ArrayList<String> arrayList) {mArrayList = arrayList;mLayoutInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);}public int getCount() {// TODO Auto-generated method stubLog.i("execute", "getCount");return mArrayList.size();}// 测试发现,getItem() 方法根本没有被调用,所以可以直接返回NULLpublic Object getItem(int arg0) {// TODO Auto-generated method stubLog.i("execute", "getItem");return null;}public long getItemId(int position) {// TODO Auto-generated method stubLog.i("execute", "getItemId");return 10;}public View getView(int position, View convertView, ViewGroup parent) {// TODO Auto-generated method stubViewHolder viewHolder = null;long start = System.currentTimeMillis();Log.i("execute", "getView:" + position);if (convertView != null) {// 发现获取值总是-1Log.i("execute", "getView id:" + convertView.getId());} else {Log.i("execute", "getView id convert is null");}if (convertView == null) {viewHolder = new ViewHolder();convertView = mLayoutInflater.inflate(R.layout.app_info_item, null);viewHolder.charTxtView = (TextView) (convertView.findViewById(R.id.app_name));  convertView.setTag(viewHolder);} else {viewHolder = (ViewHolder) convertView.getTag();}viewHolder.charTxtView.setText(mArrayList.get(position));long end = System.currentTimeMillis();long interval = end - start;Log.i("time", "time getView:" + interval);return convertView;}public static class ViewHolder {public TextView charTxtView;}}


MainActivity:

package com.test.baseadapter;import java.util.ArrayList;import android.os.Bundle;import android.app.Activity;import android.util.Log;import android.view.Menu;import android.view.View;import android.view.View.OnClickListener;import android.widget.AdapterView;import android.widget.AdapterView.OnItemClickListener;import android.widget.ListView;public class MainActivity extends Activity {ArrayList<String> arrayList = new ArrayList<String>();@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);ListView listView = (ListView) findViewById(R.id.listview);long start = getCurrentTime();// 耗时800MSfor (int i = 0; i < 10000; i++) {arrayList.add(i + "");}long end1 = getCurrentTime();long interval1 = end1 - start;Log.i("time", "interval1:" + interval1);RecentAdapter recentAdapter = new RecentAdapter(this, arrayList);long end2 = getCurrentTime();long interval2 = end2 - end1;Log.i("time", "interval2:" + interval2);listView.setAdapter(recentAdapter);long end3 = getCurrentTime();long interval3 = end3 - end2;Log.i("time", "interval3:" + interval3);listView.setOnItemClickListener(l);}// 实际上,这些VIEW的绘制是在resume之后绘制的@Overrideprotected void onResume() {// TODO Auto-generated method stubsuper.onResume();Log.i("execute", "onResume");}// 获得测试结果: position : 26 id: 10 v.getid -1OnItemClickListener l = new OnItemClickListener() {public void onClick(View v) {// TODO Auto-generated method stubLog.i("execute", "id: " + v.getId());}public void onItemClick(AdapterView<?> parent, View view, int position,long id) {// TODO Auto-generated method stubLog.i("execute", "position: " + position);Log.i("execute", "id: " + id + "  " + view.getId());}};private long getCurrentTime() {return System.currentTimeMillis();}@Overridepublic boolean onCreateOptionsMenu(Menu menu) {getMenuInflater().inflate(R.menu.activity_main, menu);return true;}}


activity_main.xml:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent" >    <ListView        android:id="@+id/listview"        android:layout_width="match_parent"        android:layout_height="match_parent" /></RelativeLayout>


app_info_item.xml:

<?xml version="1.0" encoding="utf-8"?><RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="fill_parent"    android:layout_height="wrap_content"    android:layout_gravity="center_vertical"    android:minHeight="?android:attr/listPreferredItemHeight" >    <ImageView        android:id="@+id/app_icon"        android:layout_width="@android:dimen/app_icon_size"        android:layout_height="@android:dimen/app_icon_size"        android:layout_alignParentLeft="true"        android:paddingBottom="6dip"        android:paddingLeft="6dip"        android:paddingTop="6dip"        android:scaleType="fitCenter" />    <TextView        android:id="@+id/app_name"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:layout_toRightOf="@id/app_icon"        android:paddingLeft="6dip"        android:paddingTop="6dip"        android:textAppearance="?android:attr/textAppearanceLarge"        android:textColor="?android:attr/textColorPrimary" />    <TextView        android:id="@+id/app_description"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:layout_below="@+id/app_name"        android:layout_toRightOf="@id/app_icon"        android:paddingBottom="6dip"        android:paddingLeft="6dip"        android:textAppearance="?android:attr/textAppearanceSmall" />    <TextView        android:id="@+id/app_size"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:layout_alignParentRight="true"        android:layout_below="@+id/app_name"        android:maxLines="1"        android:paddingRight="6dip"        android:textAppearance="?android:attr/textAppearanceSmall" /></RelativeLayout>


 

测试发现:
1、getCount() 和 getView(...) 方法会被反复调用,而且在刚开始加载绘制的时候getView(...) 会被多次调用,调用次数大于实际当前屏幕显示ITEM个数。至于为什么会反复调用,Google的解释,View在Draw的时候分成两个阶段:measure和layout,在measure阶段时主要就是为了计算两个参数:height和width。而且要注意的是,这是个递归的过程,从顶向下,DecorView开始依次调用自己子元素的measure。计算完成这两个参数后就开始layout,最后再是draw的调用。
对于ListView,当然每一个Item都会被调用measure方法,而在这个过程中getView和getCount会被调用,而且看用户的需求,可能会有很多次调用。而为什么会有很多组次调用呢?

问题就在于在layout中的决定ListView或者它的父元素的height和width属性的定义了。fill_parent会好一点,计算方法会比较简单,只要跟父元素的大小相似就行,但是即使是fill_parent,也不能给View当饭吃,还是要计算出来具体的dip,所以measure还是会被调用,只是可能比wrap_content的少一点。至于自适应的它会一直考量它的宽和高,根据内容(也就是它的子Item)计算宽高。
可能这个measure过程会反复执行,如果父元素也是wrap_content,这个过程会更加漫长。所以,解决方法就是尽量避免自适应,除非是万不得已,固定大小或者填充的效果会比较好一些。

解决方法: 给ListView设置固定高度或者fill_parent。


2、getItemId(int position) 方法会返回一个ID值,后续要用到的ID值就是从这个方法获得的。
比如:
 OnItemClickListener l = new OnItemClickListener() {

  public void onItemClick(AdapterView<?> parent, View view, int position,
    long id) {
   // TODO Auto-generated method stub

   Log.i("execute", "position: " + position);
   Log.i("execute", "id: " + id + "  " + view.getId());
  }
 };
onItemClick(...)方法的id就是始终从这个方法返回的id值取值的。当然,测试发现,view.getId() 返回值总为 -1, 因为这个id 是 View 的 id,和 id 是不一样的,注意不要混淆。

3、getItem(int arg0) 方法,测试发现其根本没有被系统自动调用,所以返回值完全可以返回NULL。
4、getItemId(int position),在点击相关列表项等情况会自动被调用。
5、关于 getView(int position, View convertView, ViewGroup parent) 方法:
测试发现getView(...)执行时机实际是在onResume()方法以后执行的,这个要注意。ListView 调用 setAdapter 以后,getView() 方法实际只执行差不多当前屏幕列表项次数(实际会多执行几次,但是返回的View只是当前屏幕项列表对应View)。随着屏幕上下滑动而调用getView(...)方法继续返回相应View进行绘制。
另外getView 内部写法强烈建议使用GOOGLE 官方推荐的高效方式处理,实际测试过,采用高效方法和不采用,实际每次相差都在差不多2ms 左右,这个是影响很大的。

另外 position 参数,从0 开始,一直往上递增,实际就是列表项顺序编号排序。

 

原创粉丝点击