成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Android之ListView原理學習與優化總結

移動開發 Android
利用ViewHolder來優化ListView數據加載,僅僅就此一條嗎?其實不是的,首先,想要優化ListView就得先了解ListView加載數據原理,這是前提,但是小馬在這個地方先做一些簡單的補充,大家一定仔細看下,保證會有收獲的。

在整理前幾篇文章的時候有朋友提出寫一下ListView的性能優化方面的東西,這個問題也是小馬在面試過程中被別人問到的…..今天小馬就借此機會來整理下,網上類似的資料蠻多的,倒不如自己寫一篇,記錄在這個地方,供自己以后使用,不用再翻來翻去的找了,用自己寫的…呵呵,不多講其它了,說起優化我想大家第一反應跟小馬一樣吧?想到利用ViewHolder來優化ListView數據加載,僅僅就此一條嗎?其實不是的,首先,想要優化ListView就得先了解ListView加載數據原理,這是前提,但是小馬在這個地方先做一些簡單的補充,大家一定仔細看下,保證會有收獲的:

列表的顯示需要三個元素:

  1. ListVeiw:  用來展示列表的View。

  2. 適配器 : 用來把數據映射到ListView上

  3. 數據:    具體的將被映射的字符串,圖片,或者基本組件。 

根據列表的適配器類型,列表分為三種,ArrayAdapter,SimpleAdapter和SimpleCursorAdapter,這三種適配器的使用大家可學習下官網上面的使用或者自行百度谷歌,一堆DEMO!!!其中以ArrayAdapter最為簡單,只能展示一行字。SimpleAdapter有最好的擴充性,可以自定義出各種效果。SimpleCursorAdapter可以認為是SimpleAdapter對數據庫的簡單結合,可以方便的把數據庫的內容以列表的形式展示出來。

系統要繪制ListView了,他首先用getCount()函數得到要繪制的這個列表的長度,然后開始繪制第一行,怎么繪制呢?調用getView()函數。在這個函數里面首先獲得一個View(這個看實際情況,如果是一個簡單的顯示則是View,如果是一個自定義的里面包含很多控件的時候它其實是一個ViewGroup),然后再實例化并設置各個組件及其數據內容并顯示它。好了,繪制完這一行了。那 再繪制下一行,直到繪完為止,前面這些東西做下鋪墊,繼續…….

 現在我們再來了解ListView加載數據的原理,有了這方面的了解后再說優化才行,下面先跟大家一起來看下ListView加載數據的基本原理小馬就直接寫了:

ListView的工作原理如下:

ListView 針對每個item,要求 adapter “返回一個視圖” (getView),也就是說ListView在開始繪制的時候,系統首先調用getCount()函數,根據他的返回值得到ListView的長度,然后根據這個長度,調用getView()一行一行的繪制ListView的每一項。如果你的getCount()返回值是0的話,列表一行都不會顯示,如果返回1,就只顯示一行。返回幾則顯示幾行。如果我們有幾千幾萬甚至更多的item要顯示怎么辦?為每個Item創建一個新的View?不可能!!!實際上Android早已經緩存了這些視圖,大家可以看下下面這個截圖來理解下,這個圖是解釋ListView工作原理的最經典的圖了大家可以收藏下,不懂的時候拿來看看,加深理解,其實Android中有個叫做Recycler的構件,順帶列舉下與Recycler相關的已經由Google做過N多優化過的東東比如:AbsListView.RecyclerListener、ViewDebug.RecyclerTraceType等等,要了解的朋友自己查下,不難理解,下圖是ListView加載數據的工作原理(原理圖看不清楚的點擊后看大圖):

下面簡單說下上圖的原理:

  1. 如果你有幾千幾萬甚至更多的選項(item)時,其中只有可見的項目存在內存(內存內存哦,說的優化就是說在內存中的優化!!!)中,其他的在Recycler中
  2. ListView先請求一個type1視圖(getView)然后請求其他可見的項目。convertView在getView中是空(null)的
  3. 當item1滾出屏幕,并且一個新的項目從屏幕低端上來時,ListView再請求一個type1視圖。convertView此時不是空值了,它的值是item1。你只需設定新的數據然后返回convertView,不必重新創建一個視圖
  4. 下面來看下小馬從網上找來的示例代碼,網址搞丟了,只有一個word文檔,只能 copy過來,不然直接貼網址,結合上面的原理圖一起加深理解,如下:
  1. public class MultipleItemsList extends ListActivity {    
  2.  private MyCustomAdapter mAdapter;     
  3. @Override     
  4. public void onCreate(Bundle savedInstanceState) {         
  5. super.onCreate(savedInstanceState);        
  6.  mAdapter = new MyCustomAdapter();         
  7. for (int i = 0; i < 50; i++) {            
  8.  mAdapter.addItem("item " + i);         
  9. }         
  10. setListAdapter(mAdapter);     
  11. }     
  12. private class MyCustomAdapter extends BaseAdapter {        
  13. private ArrayList mData = new ArrayList();        
  14.  private LayoutInflater mInflater;          
  15. public MyCustomAdapter() {             
  16. mInflater = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE);        
  17.  }        
  18. public void addItem(final String item) {             
  19. mData.add(item);            
  20. notifyDataSetChanged();        
  21.  }         
  22.  
  23. @Override        
  24. public int getCount() {             
  25. return mData.size();        
  26.  }        
  27.  @Override        
  28.  public String getItem(int position) {            
  29.  return mData.get(position);         
  30. }          
  31. @Override       
  32.  public long getItemId(int position) {           
  33.  return position;        
  34. }          
  35. @Override        
  36. public View getView(int position, View convertView, ViewGroup parent) {            
  37.  System.out.println("getView " + position + " " + convertView);            
  38. ViewHolder holder = null;             
  39. if (convertView == null) {                 
  40. convertView = mInflater.inflate(R.layout.item1, null);                 
  41. holder = new ViewHolder();                
  42. holder.textView = (TextView)convertView.findViewById(R.id.text);                
  43.  convertView.setTag(holder);             
  44. } else {                
  45.  holder = (ViewHolder)convertView.getTag();         
  46.    }            
  47.  holder.textView.setText(mData.get(position));            
  48.  return convertView;       
  49.  }    }    
  50.  public static class ViewHolder {        
  51.  public TextView textView;    
  52.  } }  

執行程序,查看日志:

getView 被調用 9 次 ,convertView 對于所有的可見項目是空值(如下):

 

然后稍微向下滾動List,直到item10出現:

       convertView仍然是空值,因為recycler中沒有視圖(item1的邊緣仍然可見,在頂端)再滾動列表,繼續滾動:

 

      convertView不是空值了!item1離開屏幕到Recycler中去了,然后item11被創建,再滾動下:

 

此時的convertView非空了,在item11離開屏幕之后,它的視圖(…0f8)作為convertView容納item12了,好啦,結合以上原理,下面來看看今天最主要的話題,主角ListView的優化:

             首先,這個地方先記兩個ListView優化的一個小點:

                       1. ExpandableListView 與 ListActivity 由官方提供的,里面要使用到的ListView是已經經過優化的ListView,如果大家的需求可以用Google自帶的ListView滿足的的話盡量用官方的,絕對沒錯!

                       2.其次,像小馬前面講的,說ListView優化,其實并不是指其它的優化,就是內存是的優化,提到內存…(想到OOM,折騰了我不少時間),很多很多,先來寫下,如果我們的ListView中的選項僅僅是一些簡單的TextView的話,就好辦啦,消耗不了多少的,但如果你的Item是自定義的Item的話,例如你的自定義Item布局ViewGroup中包含:按鈕、圖片、flash、CheckBox、RadioButton等一系列你能想到的控件的話, 你要在getView中單單使用文章開頭提到的ViewHolder是遠遠不夠的,如果數據過多,加載的圖片過多過大,你BitmapFactory.decode的猛多的話,OOM搞死你,這個地方再警告下大家,是警告……….也提醒下自己:

                         小馬碰到的問題大家應該也都碰到過的,自定義的ListView項亂序問題,我很天真的在getView()中強制清除了下ListView的緩存數據convertView,也就是convertView = null了,雖然當時是解決了這個問題讓其它每次重繪,但是犯了大錯了,如果數據太多的話,出現最最惡心的錯,手機卡死或強制關機,關機啊哥哥們……O_O,客戶殺了我都有可能,但大家以后別犯這樣的錯了,單單使用清除緩存convertView是解決不了實際問題的,繼續……

下面是小記:圖片用完了正確的釋放… 

  1. if(!bmp.isRecycle() ){        
  2. bmp.recycle()   //回收圖片所占的內存 
  3.  system.gc()  //提醒系統及時回收 
  4. }  

下面來列舉下真正意義上的優化吧:

  1.  ViewHolder   Tag 必不可少,這個不多說!
  2. 如果自定義Item中有涉及到圖片等等的,一定要狠狠的處理圖片,圖片占的內存是ListView項中最惡心的,處理圖片的方法大致有以下幾種:
    2.1:不要直接拿個路徑就去循環decodeFile();這是找死….用Option保存圖片大小、不要加載圖片到內存去;
    2.2:  拿到的圖片一定要經過邊界壓縮
    2.3:在ListView中取圖片時也不要直接拿個路徑去取圖片,而是以WeakReference(使用WeakReference代替強引用。比如可以使        用WeakReference mContextRef)、SoftReference、WeakHashMap等的來存儲圖片信息,是圖片信息不是圖片哦!
    2.4:在getView中做圖片轉換時,產生的中間變量一定及時釋放,用以下形式:
  3. 盡量避免在BaseAdapter中使用static 來定義全局靜態變量,我以為這個沒影響 ,這個影響很大,static是Java中的一個關鍵字,當用它來修飾成員變量時,那么該變量就屬于該類,而不是該類的實例。所以用static修飾的變量,它的生命周期是很長的,如果用它來引用一些資源耗費過多的實例(比如Context的情況最多),這時就要盡量避免使用了..
  4. 如果為了滿足需求下必須使用Context的話:Context盡量使用Application Context,因為Application的Context的生命周期比較長,引用它不會出現內存泄露的問題
  5. 盡量避免在ListView適配器中使用線程,因為線程產生內存泄露的主要原因在于線程生命周期的不可控制
  6.  記下小馬自己的錯誤:
  7. 之前使用的自定義ListView中適配數據時使用AsyncTask自行開啟線程的,這個比用Thread更危險,因為Thread只有在run函數不 結束時才出現這種內存泄露問題,然而AsyncTask內部的實現機制是運用了線程執行池(ThreadPoolExcutor,要想了解這個類的話大家加下我們的Android開發群五號,因為其它群的存儲空間快滿了,所以只上傳到五群里了,看下小馬上傳的Gallery源碼,你會對線程執行池、軟、弱、強引用有個更深入的認識),這個類產生的Thread對象的生命周期是不確定的,是應用程序無法控制的,因此如果AsyncTask作為Activity的內部類,就更容易出現內存泄露的問題。這個問題的解決辦法小馬當時網上查到了記在txt里了,如下: 
    6.1:將線程的內部類,改為靜態內部類。
    6.2:在線程內部采用弱引用保存Context引用
    示例代碼如下:
     
    1. public abstract class WeakAsyncTask extends  AsyncTask {        
    2.  protected WeakReference mTarget;            
    3. public WeakAsyncTask(WeakTarget target) {             
    4. mTarget = new WeakReference(target);         
    5. }                  
    6. @Override        
    7. protected final void onPreExecute() {            
    8. final WeakTarget target = mTarget.get();             
    9. if (target != null) {                 
    10. this.onPreExecute(target);             
    11. }        
    12.  
    13. }                   
    14.  @Override         
    15. protected final Result doInBackground(Params... params) {             
    16. final WeakTarget target = mTarget.get();            
    17. if (target != null) {                 
    18. return this.doInBackground(target, params);             
    19. } else {                
    20. return null;             
    21. }         
    22. }                    
    23. @Override        
    24. protected final void onPostExecute(Result result) {            
    25.  final WeakTarget target = mTarget.get();             
    26. if (target != null) {                 
    27. this.onPostExecute(target, result);           
    28. }        
    29. }         
    30.  protected void onPreExecute(WeakTarget target) {            
    31. // No default action         }           
    32.  protected abstract Result doInBackground(WeakTarget target, Params... params);           
    33. protected void onPostExecute(WeakTarget target, Result result) {           
    34.  // No default action         }     }  

好啦,ListVIew的優化問題,小馬就暫時先理解記錄這么多了,如果朋友們有什么更好的優化建議什么的,留言指點下小馬,一定會及時添加到進來的,先謝謝啦,其實在ListView適配器的getView()方法中可以做很多的優化,我記得還有可以優化findViewById()這個方法來尋址資源信息效率的方法,資料太多了,小馬發現了會及時更新的哦,天太晚了,先休息了,吼吼,大家加油,一起努力學習!!!O_O

責任編輯:張葉青 來源: 技術博客
相關推薦

2019-09-23 08:27:15

TCP長連接心跳

2017-12-07 15:34:57

數據庫MySQL優化原理

2015-07-03 11:12:17

產品經理技術設計篇

2013-03-27 09:17:17

Android開發AndroidList

2014-12-17 09:46:30

AndroidListView最佳實踐

2024-12-24 10:50:05

GinWeb開發

2021-07-30 19:44:51

AndroidJava線程

2009-08-31 18:34:57

C#接口事件

2011-08-25 10:07:24

Lua 5.0函數編譯器

2011-05-31 17:08:41

Android 網絡連接

2013-04-11 10:40:04

Android優化總結Android網絡編程Android常見功能

2021-07-29 14:20:34

網絡優化移動互聯網數據存儲

2014-07-29 15:57:01

ContentProv

2024-03-22 15:32:21

機器學習算法

2015-09-15 08:30:23

Android代碼優化

2019-12-13 10:25:08

Android性能優化啟動優化

2019-09-02 09:21:16

Zookeeper架構師集群

2011-05-27 15:02:15

Android ListView

2013-02-20 14:32:37

Android開發性能

2010-02-07 15:12:24

學習Android
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩免费一区二区 | 91精品国产一区二区 | caoporn免费在线视频 | 亚洲在线看 | 久久国产精品视频 | 国产综合精品一区二区三区 | 国产高清在线观看 | 99国产精品一区二区三区 | 国产成人在线视频免费观看 | 99视频免费播放 | 欧美 日韩 国产 成人 在线 | 精品一区二区三区在线观看国产 | 91亚洲精选 | 亚洲国产成人久久综合一区,久久久国产99 | 欧美成人猛片aaaaaaa | 亚洲免费av一区 | 国产精品精品视频 | 久久精品国产免费高清 | 日韩精品中文字幕在线 | av网站免费在线观看 | 99爱国产 | 理论片午午伦夜理片影院 | 国产成人一区二区三区久久久 | 国产在线精品区 | 久久精品aaa | 国产精品久久久久一区二区三区 | www.日本在线| 国产午夜精品一区二区三区四区 | 瑟瑟免费视频 | 亚洲在线高清 | 精品综合网 | 国产精品欧美一区二区 | 精品国产99 | 超碰520 | 蜜桃一区二区三区在线 | 日韩欧美国产一区二区三区 | 伊人亚洲 | www.久久 | 亚洲免费一区二区 | 操操操av| 在线观看第一页 |