編寫:allenlsy - 原文:http://developer.android.com/training/improving-layouts/smooth-scrolling.html
保持程序流暢的關鍵,是讓主線程(UI 線程)不要進行大量運算。你要確保在其他線程執行磁盤讀寫、網絡讀寫或是 SQL 操作等。為了測試你的應用的狀態,你可以啟用 StrictMode。
你應該把主線程中的耗時間的操作,提取到一個後臺線程(也叫做“worker thread工作線程”)中,使得主線程只關注 UI 繪畫。很多時候,使用 AsyncTask 是一個簡單的在主線程以外進行操作的方法。系統會自動把execute()
的請求放入隊列中併線性調用執行。這個行為是全局的,這意味著你不需要考慮自己定義線程池的事情。
在下面的例子中,一個 AsyncTask 被用於在後臺線程載入圖片,並在載入完成後把圖片顯示到 UI 上。當圖片正在載入時,它還會顯示一個進度提示。
// Using an AsyncTask to load the slow images in a background thread
new AsyncTask<ViewHolder, Void, Bitmap>() {
private ViewHolder v;
@Override
protected Bitmap doInBackground(ViewHolder... params) {
v = params[0];
return mFakeImageLoader.getImage();
}
@Override
protected void onPostExecute(Bitmap result) {
super.onPostExecute(result);
if (v.position == position) {
// If this item hasn't been recycled already, hide the
// progress and set and show the image
v.progress.setVisibility(View.GONE);
v.icon.setVisibility(View.VISIBLE);
v.icon.setImageBitmap(result);
}
}
}.execute(holder);
從 Android 3.0 (API level 11) 開始, AsyncTask 有個新特性,那就是它可以在多個 CPU 核上運行。你可以調用 executeOnExecutor()
而不是execute()
,前者可以根據CPU的核心數來觸發多個任務同時進行。
你的代碼可能在 ListView 滑動時經常使用 findViewById()
,這樣會降低性能。即使是 Adapter 返回一個用於回收的 inflate 後的視圖,你仍然需要查看這個元素並更新它。避免頻繁調用 findViewById()
的方法之一,就是使用 ViewHolder(視圖佔位符)的設計模式。
一個 ViewHolder 對象存儲了他的標籤下的每個視圖。這樣你不用頻繁查找這個元素。第一,你需要創建一個類來存儲你會用到的視圖。比如:
static class ViewHolder {
TextView text;
TextView timestamp;
ImageView icon;
ProgressBar progress;
int position;
}
然後,在 Layout 的類中生成一個 ViewHolder 對象:
ViewHolder holder = new ViewHolder();
holder.icon = (ImageView) convertView.findViewById(R.id.listitem_image);
holder.text = (TextView) convertView.findViewById(R.id.listitem_text);
holder.timestamp = (TextView) convertView.findViewById(R.id.listitem_timestamp);
holder.progress = (ProgressBar) convertView.findViewById(R.id.progress_spinner);
convertView.setTag(holder);
這樣你就可以輕鬆獲取每個視圖,而不是使用 findViewById()
來不斷查找子視圖,節省了寶貴的運算時間。