ListView BaseAdapter 簡單分析

接觸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 stub

		Log.i("execute", "getCount");

		return mArrayList.size();
	}

	// 測試發現,getItem() 方法根本沒有被調用,所以可以直接返回NULL
	public Object getItem(int arg0) {
		// TODO Auto-generated method stub

		Log.i("execute", "getItem");

		return null;
	}

	public long getItemId(int position) {
		// TODO Auto-generated method stub

		Log.i("execute", "getItemId");

		return 10;
	}

	public View getView(int position, View convertView, ViewGroup parent) {
		// TODO Auto-generated method stub

		ViewHolder viewHolder = null;

		long start = System.currentTimeMillis();

		Log.i("execute", "getView:" + position);

		if (convertView != null) {
			// 發現獲取值總是-1
			Log.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>();

	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);

		ListView listView = (ListView) findViewById(R.id.listview);

		long start = getCurrentTime();

		// 耗時800MS
		for (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之後繪製的
	@Override
	protected void onResume() {
		// TODO Auto-generated method stub
		super.onResume();

		Log.i("execute", "onResume");
	}

	// 獲得測試結果: position : 26 id: 10 v.getid -1
	OnItemClickListener l = new OnItemClickListener() {

		public void onClick(View v) {
			// TODO Auto-generated method stub
			Log.i("execute", "id: " + v.getId());
		}

		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());
		}
	};

	private long getCurrentTime() {
		return System.currentTimeMillis();
	}

	@Override
	public 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 開始,一直往上遞增,實際就是列表項順序編號排序。

 

發佈了24 篇原創文章 · 獲贊 1 · 訪問量 1萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章