Android之史上最強ListView優化提案

Android之史上最強ListView優化提案

www.MyException.Cn  網友分享於:2015-08-04  瀏覽:0次android

Android之史上最強ListView優化方案緩存

在android開發中Listview是一個很重要的組件,它以列表的形式根據數據的長自適應展現具體內容,用戶能夠自由的定義listview每一列的佈局,但當listview有大量的數據須要加載的時候,會佔據大量內存,影響性能。

 

本文的重點便是從以下幾個方面介紹如何對ListView進行優化。

 

一、convertView重用

 

Android SDK中這樣講:網絡

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 viewapp

利用好 convertView 來重用 View,切忌每次 getView() 都新建。ListView 的核心原理就是重用 View,若是重用 view 不改變寬高,重用View能夠減小從新分配緩存形成的內存頻繁分配/回收;框架

 

<?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/listview"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:cacheColorHint="#00000000" >
</ListView>
</LinearLayout>

ListView的android:layout_height屬性值設置爲"fill_parent"或者''wrap_content"狀況不同,可是convertView的機制同樣iview

 

若是設置爲fill_parent:屏幕上顯示出的Item的convertview都爲空,向下滑動新產生的Item的convetview都不爲空異步

若是設置爲wrap_content:只有第一個Item的convertview爲null,其餘的不爲空ide

總結:佈局

在初始顯示的時候,每次顯示一個item都調用一次getview方法可是每次調用的時候covertview爲空(由於尚未舊的view),當顯示完 了以後。若是屏幕移動了以後,而且致使有些Item(也能夠說是view)跑到屏幕外面,此時若是還有新的item須要產生,則這些item顯示時調用的 getview方法中的convertview參數就不是null,而是那些移出屏幕的view(舊view),咱們所要作的就是將須要顯示的item填 充到這些回收的view(舊view)中去,最後注意convertview爲null的不只僅是初始顯示的那些item,還有一些是已經開始移入屏幕但 是尚未view被回收的那些item。
 性能

二、ViewHolder優化

 

使用ViewHolder的緣由是findViewById方法耗時較大,若是控件個數過多,會嚴重影響性能,而使用ViewHolder主要是爲了能夠省去這個時間。經過setTag,getTag直接獲取View

總結:

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取出加上新的數據就好了

 

class  ViewHolder{
    ImageView img;
    TextView name;
}

public View getView(int position, View convertView, ViewGroup parent) {
    ViewHolder holder = null;
    if(convertView==null){
        convertView = inflater.inflate(R.layout.list_item, parent, false);
        holder.img = (ImageView) convertView.findViewById(R.id.img);
        holder.name = (TextView) convertView.findViewById(R.id.name);
        holder = new ViewHolder();
        convertView.setTag(holder);
    }else{
        holder = (ViewHolder) convertView.getTag();
    }
    //設置holder
    holder.img.setImageResource(R.drawable.ic_launcher);
    holder.name.setText(list.get(position).partname);
    return convertView;
}

 

三、圖片加載優化

 

若是ListView須要加載顯示網絡圖片,咱們儘可能不要在ListView滑動的時候加載圖片,那樣會使ListView變得卡頓,因此咱們 須要在監聽器裏面監聽ListView的狀態,若是ListView滑動(SCROLL_STATE_TOUCH_SCROLL)或者被猛滑 (SCROLL_STATE_FLING)的時候,中止加載圖片,若是沒有滑動(SCROLL_STATE_IDLE),則開始加載圖片。

 

假如咱們要本身實現應該怎麼作那,這裏提供個思路

/** 
     * list滾動監聽 
     */  
    listView.setOnScrollListener(new OnScrollListener() {  
        @Override  
        public void onScrollStateChanged(AbsListView view, int scrollState) {  
            
            if (scrollState == OnScrollListener.SCROLL_STATE_IDLE) {//list中止滾動時加載圖片  
                loadImage(startPos, endPos);// 異步加載圖片   ,只加載能夠看到的圖片  
            }  
        }  
        @Override  
        public void onScroll(AbsListView view, int firstVisibleItem,  
                int visibleItemCount, int totalItemCount) {  
            //設置當前屏幕顯示的起始pos和結束pos 
        	startPos = firstVisibleItem;  
        	endPos = firstVisibleItem + visibleItemCount;  
            if (endPos >= totalItemCount) {  
            	endPos = totalItemCount - 1;  
            }  
        }  
    });

 

其實在Universal-Image-loader框架中就存在這個功能,並且作的很好,徹底能夠直接拿來使用,代碼中咱們一般這樣設置:

listView = (ListView) rootView.findViewById(R.id.fragment_user_info_lisiview);
 listView.setOnScrollListener(DisplayImageOptionsUtil.getPauseOnScrollListener(this));
 listView.setOnItemClickListener(this);
public static PauseOnScrollListener getPauseOnScrollListener(OnScrollListener scrollListener) {
        PauseOnScrollListener listener = new PauseOnScrollListener(ImageLoader.getInstance(),
                false, true, scrollListener);
        return listener;
    }

PauseOnScrollListener的第一個參數指的是圖片加載對象ImageLoader,第二個參數爲pauseOnScroll來控制是否在滑動的過程當中暫停加載圖片,若是須要暫停則傳true,第三個參數控制猛的滑動界面的時候圖片是否加載。

打開PauseOnScrollListener的源碼,咱們能夠看到,在listview滑動或者被猛一下滑動的時候,調用了imageLoader.pause()方法

/**
	 * Constructor
	 *
	 * @param imageLoader    {@linkplain ImageLoader} instance for controlling
	 * @param pauseOnScroll  Whether {@linkplain ImageLoader#pause() pause ImageLoader} during touch scrolling
	 * @param pauseOnFling   Whether {@linkplain ImageLoader#pause() pause ImageLoader} during fling
	 * @param customListener Your custom {@link OnScrollListener} for {@linkplain AbsListView list view} which also
	 *                       will be get scroll events
	 */
	public PauseOnScrollListener(ImageLoader imageLoader, boolean pauseOnScroll, boolean pauseOnFling,
			OnScrollListener customListener) {
		this.imageLoader = imageLoader;
		this.pauseOnScroll = pauseOnScroll;
		this.pauseOnFling = pauseOnFling;
		externalListener = customListener;
	}

	@Override
	public void onScrollStateChanged(AbsListView view, int scrollState) {
		switch (scrollState) {
			case OnScrollListener.SCROLL_STATE_IDLE:
				imageLoader.resume();
				break;
			case OnScrollListener.SCROLL_STATE_TOUCH_SCROLL:
				if (pauseOnScroll) {
					imageLoader.pause();
				}
				break;
			case OnScrollListener.SCROLL_STATE_FLING:
				if (pauseOnFling) {
					imageLoader.pause();
				}
				break;
		}
		if (externalListener != null) {
			externalListener.onScrollStateChanged(view, scrollState);
		}
	}

 

四、onClickListener處理

當ListView的item中有好比button這些子view時,須要對其設置onclickListener,一般的寫法是在getView方法中一個個設置,好比

holder.img.setonClickListener(new onClickListenr)...

可是這種寫法每次調用getView時都設置了一個新的onClick事件,效率很低。高效的寫法能夠直接在ViewHolder中設置一個position,而後viewHolder implements OnClickListenr:

class  ViewHolder implements OnClickListener{
    int position;
    TextView name;

    public void setPosition(int position){
        this.position = position;
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()){
            //XXXX
        }
    }
}

public View getView(int position, View convertView, ViewGroup parent) {
    ViewHolder holder = null;
    if(convertView==null){
        convertView = inflater.inflate(R.layout.list_item, parent, false);
        holder = new ViewHolder();
        holder.name = (TextView) convertView.findViewById(R.id.name);
        holder.name.setOnClickListener(this);
        convertView.setTag(holder);
    }else{
        holder = (ViewHolder) convertView.getTag();
    }
    //設置holder
    holder.name.setText(list.get(position).partname);
    //設置position
    holder.setPosition(position);
    return convertView;
}


補充:ListView的listitem裏面含有Button  CheckBox之類的子控件的時候,子控件會把Focus搶去,最簡單有效的解決方法是在ListView的item佈局文件根元素中設置屬 性  android:descendantFocusability="blocksDescendants"

 

五、減小Item View的佈局層級

這是全部layout都必須遵循的,佈局層級過深會直接致使View的測量與繪製浪費大量的時間

 

六、adapter中的getView方法儘可能少使用邏輯

不要在getView方法中作過於複雜的邏輯,能夠想辦法抽離到別的地方,舉個例子

優化前的getView():

@Override
public View getView(int position, View convertView, ViewGroup paramViewGroup) {
        Object current_event = mObjects.get(position);
        ViewHolder holder = null;
        if (convertView == null) {
                holder = new ViewHolder();
                convertView = inflater.inflate(R.layout.row_event, null);
                holder.ThreeDimension = (ImageView) convertView.findViewById(R.id.ThreeDim);
                holder.EventPoster = (ImageView) convertView.findViewById(R.id.EventPoster);
                convertView.setTag(holder);

        } else {
                holder = (ViewHolder) convertView.getTag();
        }

       //在這裏進行邏輯判斷,這是有問題的 
        if (doesSomeComplexChecking()) {
                holder.ThreeDimention.setVisibility(View.VISIBLE);
        } else {
                holder.ThreeDimention.setVisibility(View.GONE); 
        }

        // 這是設置image的參數,每次getView方法執行時都會執行這段代碼,這顯然是有問題的
        RelativeLayout.LayoutParams imageParams = new RelativeLayout.LayoutParams(measuredwidth, rowHeight);
        holder.EventPoster.setLayoutParams(imageParams);

        return convertView;
}

優化後的getView():

@Override
public View getView(int position, View convertView, ViewGroup paramViewGroup) {
    Object object = mObjects.get(position);
    ViewHolder holder = null;

    if (convertView == null) {
            holder = new ViewHolder();
            convertView = inflater.inflate(R.layout.row_event, null);
            holder.ThreeDimension = (ImageView) convertView.findViewById(R.id.ThreeDim);
            holder.EventPoster = (ImageView) convertView.findViewById(R.id.EventPoster);
            //設置參數提到這裏,只有第一次的時候會執行,以後會複用 
            RelativeLayout.LayoutParams imageParams = new RelativeLayout.LayoutParams(measuredwidth, rowHeight);
            holder.EventPoster.setLayoutParams(imageParams);
            convertView.setTag(holder);
    } else {
            holder = (ViewHolder) convertView.getTag();
    }

    // 咱們直接經過對象的getter方法代替剛纔那些邏輯判斷,那些邏輯判斷放到別的地方去執行了
    holder.ThreeDimension.setVisibility(object.getVisibility());

    return convertView;
}

七、adapter中的getView方法儘可能少作耗時操做

八、adapter中的getView方法避免建立大量對象

九、將ListView的scrollingCache和animateCache設置爲false

這兩個屬性,默認狀況下是開啓的,會消耗大量的內存,所以會頻繁調用GC,咱們能夠手動將它關閉掉(視狀況而定)

 

其它

 

一、利用好 View Type,例如你的 ListView 中有幾個類型的 Item,須要給每一個類型建立不一樣的 View,這樣有利於 ListView 的回收,固然類型不能太多

二、善用自定義 View,自定義 View 能夠有效的減少 Layout 的層級,並且對繪製過程能夠很好的控制;

三、儘可能能保證 Adapter 的 hasStableIds() 返回 true,這樣在 notifyDataSetChanged() 的時候,若是 id 不變,ListView 將不會從新繪製這個 View,達到優化的目的;

四、每一個Item 不能過高,特別是不要超過屏幕的高度,能夠參考 Facebook 的優化方法,把特別複雜的 Item 分解成若干小的 Item

五、ListView 中元素避免半透明

六、儘可能開啓硬件加速

七、使用 RecycleView 代替。 ListView 每次更新數據都要 notifyDataSetChanged(),有些太暴力了。RecycleView 在性能和可定製性上都有很大的改善,推薦使用。

相關文章
相關標籤/搜索