BaseAdapter的深度學習
博主工作了幾年,也用了幾年的ListView等AdapterView控件,但關於Adapter的一些問題並沒有深入下去,終於有時間學習總結下關於BaseAdapter的一些較深入的話題。本文涉及三個話題:Adapter的回收機制和效率提升,getItemViewType()/getViewTypeCount()方法以及notifyDatasetChanged()使用的注意點。
1.Adapter的回收機制和效率提升
Android在繪制Adapter時,系統首先調用getCount()方法,根據它的返回值得到ListView的長度,然後根據這個長度,調用getView()方法逐行繪制。如果ListView的長度超過了屏幕的長度,android只會繪制顯示出來的Item,同時,系統會回收走隱藏的Item。
如下圖所示,此時系統繪制的只有position:4到positon12這9個Item.若按箭頭方法滑動,將回收position12,以及繪制position3.
總的來說,顯示出來然後因為拖動而被隱藏的Item才會觸發回收。在方法getView(int position, View convertView, ViewGroup parent)中,第二個參數convertView的含義:是代表系統最近回收的View。若整屏能顯示9個Item,第一次打開帶ListView的控件時,因為並沒有回收的View,調用getVIew時,參數convertView的值會為null,否則將不是null,而是最近回收的View的引用.那麼合理利用convertView將是提升Adapter效率的關鍵,否則將會產生大量的new View開銷。
復制代碼
1 @Override
2 public View getView(int position, View convertView, ViewGroup parent)
3 {
4 Holder1 holder1 = null;
5 if(null==convertView)
6 {
7 System.out.println("convertView == null" + " position:" + position);
8 holder1=new Holder1();
9 convertView=LayoutInflater.from(mContext).inflate(R.layout.textview, null);
10 holder1.textView=(TextView)convertView.findViewById(R.id.textview);
11 convertView.setTag(holder1);
12 }
13 else
14 {
15 holder1=(Holder1)convertView.getTag();
16 System.out.println("重用:" + holder1.textView.getText());
17 }
18 holder1.textView.setText("position: "+position);
19 return convertView;
20 }
21
22 class Holder1
23 {
24 public TextView textView;
25 }
復制代碼
說明一下上圖中的例子,按箭頭方法拖動,接下來將顯示position=4的Item,系統調用getView方法時,第二個參數convertView的值將是position=12的View的引用(最近回收的一個Item的View).[讀者可在convertView中用一個TextView記錄下每個View的position值,就可發現這個規律]
精致的邏輯說明:系統繪制Item的View和回收Item的View時有個規則:該Item只要顯示出一點點就觸發繪制,但必須等該Item完全隱藏之後才觸發回收。試驗上例時若結果對不上請注意這條說明。