[轉載]Android利用convertView最佳化ListView效能

來源:互聯網
上載者:User

標籤:

本的getView寫法

Java代碼
public View getView(int position, View convertView, ViewGroup parent) {
View view = new View();

//通過inflate等找到布局 然後findViewById等 設定各個顯示的item

return view;
}

而在ListView滑動的過程中 很容易就會發現每次getView被執行 都會new出一個View對象 長此以往會產生很大的消耗 
特別當item中還有Bitmap等 甚至會造成OOM的錯誤導致程式崩潰

在看getView提供的參數時 可能已經注意到了 有一個參數View convertView 而這個convertView其實就是最關鍵的部分了 原理上講 
當ListView滑動的過程中 會有item被滑出螢幕 而不再被使用 這時候Android會回收這個條目的view 
這個view也就是這裡的convertView
在上面的做法中 當item1被移除螢幕的時候 我們會重新new一個View給新顯示的item_new 而如果使用了這個convertView 我們其實可以複用它 
這樣就省去了new View的大量開銷


下面就是使用convertView後的情況

Java代碼
public View getView(int position, View convertView, ViewGroup parent) {
View view = null;
if (convertView != null) {
view = convertView;
//複用了回收的view 只需要直接作內容填充的修改就好了
} else {
view = new Xxx(...);
//沒有供複用的view 按一般的做法建立view
}
return view;
}

這樣一來 就避免了反覆建立大量view的問題了

但是上面的仍然有缺陷 當我們的ListView中填充的item有多種形式時 比如微博中 有的item中包含圖片 有的item包含視頻 那麼必然的 
我們需要用到2種item的布局方式
此時如果只是單純判斷convert是否存在 會造成回收的view不符合你當前需要的布局 而類似轉換失敗出錯退出
這裡要提到Adapter中的另外2個方法:


public int getItemViewType(int position) {}
public int getViewTypeCount() {}


從方法名上 就可以比較明顯的明白這2個的作用
下面附上一個demo代碼
Java代碼

class MyAdapter extends BaseAdapter{
Context mContext;
LinearLayout linearLayout = null;
LayoutInflater inflater;
TextView tex;
final int VIEW_TYPE = 2;
final int TYPE_1 = 0;
final int TYPE_2 = 1;

public MyAdapter(Context context) {
mContext = context;
inflater = LayoutInflater.from(mContext);
}

@Override
public int getCount() {
return listString.size();
}

//每個convert view都會調用此方法,獲得當前所需要的view樣式
@Override
public int getItemViewType(int position) {
int p = position%6;
if(p == 0)
return TYPE_1;
else if(p < 3)
return TYPE_2;
else
return TYPE_1;
}

@Override
public int getViewTypeCount() {
return 2;
}

@Override
public Object getItem(int arg0) {
return listString.get(arg0);
}

@Override
public long getItemId(int position) {
return position;
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
viewHolder1 holder1 = null;
viewHolder2 holder2 = null;
int type = getItemViewType(position);

//無convertView,需要new出各個控制項
if(convertView == null)
{
//按當前所需的樣式,確定new的布局
switch(type)
{
case TYPE_1:
convertView = inflater.inflate(R.layout.listitem1, parent, false);
holder1 = new viewHolder1();
holder1.textView = (TextView)convertView.findViewById(R.id.textview1);
holder1.checkBox = (CheckBox)convertView.findViewById(R.id.checkbox);
convertView.setTag(holder1);
break;
case TYPE_2:
convertView = inflater.inflate(R.layout.listitem2, parent, false);
holder2 = new viewHolder2();
holder2.textView = (TextView)convertView.findViewById(R.id.textview2);
holder2.imageView = (ImageView)convertView.findViewById(R.id.imageview);
convertView.setTag(holder2);
break;
}
}
else
{
//有convertView,按樣式,取得不用的布局
switch(type)
{
case TYPE_1:
holder1 = (viewHolder1) convertView.getTag();
break;
case TYPE_2:
holder2 = (viewHolder2) convertView.getTag();
break;
}
//設定資源
switch(type)
{
case TYPE_1:
holder1.textView.setText(Integer.toString(position));
holder1.checkBox.setChecked(true);
break;
case TYPE_2:
holder2.textView.setText(Integer.toString(position));
holder2.imageView.setBackgroundResource(R.drawable.icon);
break;
}
}
return convertView;
}
}
//各個布局的控制項資源
class viewHolder1{
CheckBox checkBox;
TextView textView;
}
class viewHolder2{
ImageView imageView;
TextView textView;
}

這裡對於每個View使用了一個viewHolder來控制其內部的子item
還有一個需要注意的地方是使用了setTag和getTag的方法 將holder綁定到了view上 也算一種技巧

以上基本就是主要的內容了 下面再補充實際操作當中的一些Tips
*如果convertView上用Type區分有些繁瑣 或者不需要那麼複雜 只是很少有出現不同的情況 那麼還可以在取得convertView後 通過java提供的 
instanceof 來判斷是否可以強轉 如果不能強轉 就去建立一個View的做法 但是其實這種做法並不規範 所以還是推薦上面的做法
*第二個是關於ListView 對於純色的item背景 其實可以直接設定BackgroundColor 而不要使用圖片 這一部分其實可以有不小的提升 同樣的 
對於任何純色的背景 應該盡量去設定RGB顏色 而不是全用一張圖片做背景返回 

[轉載]Android利用convertView最佳化ListView效能

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.