欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Android開發(fā)中ListView控件性能的一些優(yōu)化方法

 更新時間:2016年01月18日 09:37:18   作者:左老怪  
這篇文章主要介紹了Android開發(fā)中ListView控件性能的一些優(yōu)化方法,需要的朋友可以參考下

ListView優(yōu)化一直是一個老生常談的問題,不管是面試還是平常的開發(fā)中,ListView永遠(yuǎn)不會被忽略掉,那么這篇文章我們來看看如何最大化的優(yōu)化ListView的性能。

1.在adapter中的getView方法中盡量少使用邏輯

2.盡最大可能避免GC

3.滑動的時候不加載圖片

4.將ListView的scrollingCache和animateCache設(shè)置為false

5.item的布局層級越少越好

6.使用ViewHolder

下面就具體來看一些

1.在adapter中的getView方法中盡量少使用邏輯

不要在你的getView()中寫過多的邏輯代碼,我們可以將這些代碼放在別的地方,例如:

優(yōu)化前的getView():

@Override

public View getView(intposition, View convertView, ViewGroup paramViewGroup) {

Object current_event = mObjects.get(position);

ViewHolder holder =null;if(convertView ==null) {

holder =newViewHolder();

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

}

//在這里進(jìn)行邏輯判斷,這是有問題的

if(doesSomeComplexChecking()) {

holder.ThreeDimention.setVisibility(View.VISIBLE);

}else{

holder.ThreeDimention.setVisibility(View.GONE);

}

// 這是設(shè)置image的參數(shù),每次getView方法執(zhí)行時都會執(zhí)行這段代碼,這顯然是有問題的

RelativeLayout.LayoutParams imageParams =newRelativeLayout.LayoutParams(measuredwidth, rowHeight);

holder.EventPoster.setLayoutParams(imageParams);

returnconvertView;

}

優(yōu)化后的getView():

@Override

public View  getView(intposition, View convertView, ViewGroup paramViewGroup) {

Object object = mObjects.get(position);

ViewHolder holder =null;if(convertView ==null) {

holder =newViewHolder();

convertView = inflater.inflate(R.layout.row_event,null);

holder.ThreeDimension = (ImageView) convertView.findViewById(R.id.ThreeDim);

holder.EventPoster = (ImageView) convertView.findViewById(R.id.EventPoster);

//設(shè)置參數(shù)提到這里,只有第一次的時候會執(zhí)行,之后會復(fù)用

RelativeLayout.LayoutParams imageParams =newRelativeLayout.LayoutParams(measuredwidth, rowHeight);

holder.EventPoster.setLayoutParams(imageParams);

convertView.setTag(holder);

}else{

holder = (ViewHolder) convertView.getTag();

}// 我們直接通過對象的getter方法代替剛才那些邏輯判斷,那些邏輯判斷放到別的地方去執(zhí)行了holder.ThreeDimension.setVisibility(object.getVisibility());returnconvertView;

}

2.GC 垃圾回收器

當(dāng)你創(chuàng)建了大量的對象的時候,GC就會頻繁的執(zhí)行,所以在getView()方法中不要創(chuàng)建很多的對象,最好的優(yōu)化是,不要在ViewHolder以外創(chuàng)建任何對象,如果你的你的log里面發(fā)現(xiàn)“GC has freed some memory”頻繁出現(xiàn)的話,那你的程序肯定有問題了。你可以檢查一下:

a) item布局的層級是否太深

b) getView()方法中是否有大量對象存在

c) ListView的布局屬性

3.加載圖片

如果你的ListView中需要顯示從網(wǎng)絡(luò)上下載的圖片的話,我們不要在ListView滑動的時候加載圖片,那樣會使ListView變得卡頓,所以我們需要再監(jiān)聽器里面監(jiān)聽ListView的狀態(tài),如果滑動的時候,停止加載圖片,如果沒有滑動,則開始加載圖片

listView.setOnScrollListener(newOnScrollListener() {

@Override

public void onScrollStateChanged(AbsListView listView,intscrollState) {

//停止加載圖片

if(scrollState == AbsListView.OnScrollListener.SCROLL_STATE_FLING) {

imageLoader.stopProcessingQueue();

}else{

//開始加載圖片

imageLoader.startProcessingQueue();

}

}

@Override

public void onScroll(AbsListView view,intfirstVisibleItem,intvisibleItemCount,inttotalItemCount) {

// TODO Auto-generated method stub}

});

4.將ListView的scrollingCache和animateCache設(shè)置為false

scrollingCache:scrollingCache本質(zhì)上是drawing cache,你可以讓一個View將他自己的drawing保存在cache中(保存為一個bitmap),這樣下次再顯示View的時候就不用重畫了,而是從cache中取出。默認(rèn)情況下drawing cahce是禁用的,因?yàn)樗膬?nèi)存了,但是它確實(shí)比重畫來的更加平滑。而在ListView中,scrollingCache是默認(rèn)開啟的,我們可以手動將它關(guān)閉。

animateCache:ListView默認(rèn)開啟了animateCache,這會消耗大量的內(nèi)存,因此會頻繁調(diào)用GC,我們可以手動將它關(guān)閉掉

優(yōu)化前的ListView

<android:id="@android:id/list"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:cacheColorHint="#00000000"

android:divider="@color/list_background_color"

android:dividerHeight="0dp"

android:listSelector="#00000000"

android:smoothScrollbar="true"

android:visibility="gone"/>

優(yōu)化后的ListView

<android:id="@android:id/list"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:divider="@color/list_background_color"

android:dividerHeight="0dp"

android:listSelector="#00000000"

android:scrollingCache="false"

android:animationCache="false"

android:smoothScrollbar="true"

android:visibility="gone"/>

5.減少item的布局的深度

我們應(yīng)該盡量減少item布局深度,因?yàn)楫?dāng)滑動ListView的時候,這回直接導(dǎo)致測量與繪制,因此會浪費(fèi)大量的時間,所以我們應(yīng)該將一些不必要的布局嵌套關(guān)系去掉。減少item布局深度

6.使用ViewHolder

這個大家應(yīng)該非常熟悉了,但是不要小看這個ViewHolder,它可以大大提高我們ListView的性能

ListView的優(yōu)化我們已經(jīng)講完了,如果在你的項(xiàng)目中,這些基本優(yōu)化你還沒有做到的話,那么你的ListView是有問題的,還有很大的提升潛力,以后再使用ListView的時候,一定要將這幾點(diǎn)考慮進(jìn)去,發(fā)揮它的最大的性能。

附:

此外,如果發(fā)現(xiàn)性能方面的問題,以下幾個常見問題也應(yīng)注意:

1..Adapter的getView方法里面convertView沒有使用setTag和getTag方式;

2.在getView方法里面ViewHolder初始化后的賦值或者是多個控件的顯示狀態(tài)和背景的顯示沒有優(yōu)化好,抑或是里面含有復(fù)雜的計算和耗時操作;

3.在getView方法里面 inflate的row 嵌套太深(布局過于復(fù)雜)或者是布局里面有大圖片或者背景所致;

4.Adapter多余或者不合理的notifySetDataChanged;

5.listview 被多層嵌套,多次的onMessure導(dǎo)致卡頓,如果多層嵌套無法避免,建議把listview的高和寬設(shè)置為fill_parent. 如果是代碼繼承的listview,那么也請你別忘記為你的繼承類添加上LayoutPrams,注意高和寬都是fill_parent的;  以往我一般都是將listview的高度設(shè)置成fill_parent,而這次我是設(shè)為wrap_content,這樣做的問題在于,ListView沒有取到實(shí)際的高度,他還要根據(jù)計算才能確定,而每一次計算應(yīng)該會觸發(fā)listview的渲染,所以就會出現(xiàn)getview的調(diào)用次數(shù)跟正常情況相比多了好幾倍。所以在一般情況下,我建議把listiview在布局文件中的高度總是設(shè)置為:fill_parent(或者match_parent),這不僅僅是getview的調(diào)用次數(shù)問題,還涉及到布局的效率。

相關(guān)文章

最新評論