基本的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顏色 而不是全用一張圖片做背景返回 |