淺談Android開發(fā)中ListView控件性能的一些優(yōu)化方法
ListView優(yōu)化一直是一個老生常談的問題,不管是面試還是平常的開發(fā)中,ListView永遠不會被忽略掉,那么這篇文章我們來看看如何最大化的優(yōu)化ListView的性能。
1.在adapter中的getView方法中盡量少使用邏輯
2.盡最大可能避免GC
3.滑動的時候不加載圖片
4.將ListView的scrollingCache和animateCache設置為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(); } //在這里進行邏輯判斷,這是有問題的 if(doesSomeComplexChecking()) { holder.ThreeDimention.setVisibility(View.VISIBLE); }else{ holder.ThreeDimention.setVisibility(View.GONE); } // 這是設置image的參數,每次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); //設置參數提到這里,只有第一次的時候會執(zhí)行,之后會復用 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 垃圾回收器
當你創(chuàng)建了大量的對象的時候,GC就會頻繁的執(zhí)行,所以在getView()方法中不要創(chuàng)建很多的對象,最好的優(yōu)化是,不要在ViewHolder以外創(chuàng)建任何對象,如果你的你的log里面發(fā)現“GC has freed some memory”頻繁出現的話,那你的程序肯定有問題了。你可以檢查一下:
a) item布局的層級是否太深
b) getView()方法中是否有大量對象存在
c) ListView的布局屬性
3.加載圖片
如果你的ListView中需要顯示從網絡上下載的圖片的話,我們不要在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設置為false
scrollingCache:scrollingCache本質上是drawing cache,你可以讓一個View將他自己的drawing保存在cache中(保存為一個bitmap),這樣下次再顯示View的時候就不用重畫了,而是從cache中取出。默認情況下drawing cahce是禁用的,因為它太耗內存了,但是它確實比重畫來的更加平滑。而在ListView中,scrollingCache是默認開啟的,我們可以手動將它關閉。
animateCache:ListView默認開啟了animateCache,這會消耗大量的內存,因此會頻繁調用GC,我們可以手動將它關閉掉
優(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的布局的深度
我們應該盡量減少item布局深度,因為當滑動ListView的時候,這回直接導致測量與繪制,因此會浪費大量的時間,所以我們應該將一些不必要的布局嵌套關系去掉。減少item布局深度
6.使用ViewHolder
這個大家應該非常熟悉了,但是不要小看這個ViewHolder,它可以大大提高我們ListView的性能
ListView的優(yōu)化我們已經講完了,如果在你的項目中,這些基本優(yōu)化你還沒有做到的話,那么你的ListView是有問題的,還有很大的提升潛力,以后再使用ListView的時候,一定要將這幾點考慮進去,發(fā)揮它的最大的性能。
附:
此外,如果發(fā)現性能方面的問題,以下幾個常見問題也應注意:
1..Adapter的getView方法里面convertView沒有使用setTag和getTag方式;
2.在getView方法里面ViewHolder初始化后的賦值或者是多個控件的顯示狀態(tài)和背景的顯示沒有優(yōu)化好,抑或是里面含有復雜的計算和耗時操作;
3.在getView方法里面 inflate的row 嵌套太深(布局過于復雜)或者是布局里面有大圖片或者背景所致;
4.Adapter多余或者不合理的notifySetDataChanged;
5.listview 被多層嵌套,多次的onMessure導致卡頓,如果多層嵌套無法避免,建議把listview的高和寬設置為fill_parent. 如果是代碼繼承的listview,那么也請你別忘記為你的繼承類添加上LayoutPrams,注意高和寬都是fill_parent的; 以往我一般都是將listview的高度設置成fill_parent,而這次我是設為wrap_content,這樣做的問題在于,ListView沒有取到實際的高度,他還要根據計算才能確定,而每一次計算應該會觸發(fā)listview的渲染,所以就會出現getview的調用次數跟正常情況相比多了好幾倍。所以在一般情況下,我建議把listiview在布局文件中的高度總是設置為:fill_parent(或者match_parent),這不僅僅是getview的調用次數問題,還涉及到布局的效率。
- android ListView和ProgressBar(進度條控件)的使用方法
- Android控件之ListView用法實例詳解
- Android ExpandableListView展開列表控件使用實例
- Android控件ListView用法(讀取聯(lián)系人示例代碼)
- Android UI控件ExpandableListView基本用法詳解
- android使用ExpandableListView控件實現小說目錄效果的例子
- Android控件之使用ListView實現時間軸效果
- Android 自定義彈性ListView控件實例代碼(三種方法)
- Android編程之控件ListView使用方法
- Android控件ListView使用方法詳解
相關文章
Android 開發(fā)中根據搜索內容實現TextView中的文字部分加粗
最近遇到一個需求,需要做一個搜索功能。搜索的內容需要加粗顯示。實現方法很簡單,下面通過本文給大家分享Android 開發(fā)中根據搜索內容實現TextView中的文字部分加粗樣式,非常不錯,需要的朋友參考下2017-03-03Android編程圖片加載類ImageLoader定義與用法實例分析
這篇文章主要介紹了Android編程圖片加載類ImageLoader定義與用法,結合實例形式分析了Android圖片加載類ImageLoader的功能、定義、使用方法及相關操作注意事項,代碼中備有較為詳盡的注釋便于理解,需要的朋友可以參考下2017-12-12