导语:在开发Android应用的过程当中,咱们须要时刻注意保障应用的稳定性和界面响应性,由于不稳定或者响应速度慢的应用将会给用户带来很是差的交互体验。在愈来愈讲究用户体验的大环境下,用户也许会由于应用的一次Force Close(简称FC)或者延迟严重的动画效果而卸载你的应用。因为如今的应用大多须要异步链接网络,本系列文章就以构建网络应用为例,从稳定性和响应性两个角度分析多线程网络任务的性能优化方法。java
概述:为了避免阻塞UI线程(亦称主线程),提升应用的响应性,咱们常常会使用新开线程的方式,异步处理那些致使阻塞的任务(如要了解Android异步处理的实现方式和原理,请先阅读《Android异步处理系列文章索引》)。android
AsyncTask是Android为咱们提供的方便编写异步任务的工具类,可是,在了解AsyncTask的实现原理以后,发现AsyncTask并不能知足咱们全部的需求,使用不当还有可能致使应用FC。性能优化
本文主要经过分析AsyncTask提交任务的策略和一个具体的例子,说明AsyncTask的不足之处,至于解决办法,咱们将在下篇再讲解。网络
分析:多线程
AsyncTask类包含一个全局静态的线程池,线程池的配置参数以下:app
private static final int CORE_POOL_SIZE =5;//5个核心工做线程 private static final int MAXIMUM_POOL_SIZE = 128;//最多128个工做线程 private static final int KEEP_ALIVE = 1;//空闲线程的超时时间为1秒 private static final BlockingQueue<Runnable> sWorkQueue = new LinkedBlockingQueue<Runnable>(10);//等待队列 private static final ThreadPoolExecutorsExecutor = new ThreadPoolExecutor(CORE_POOL_SIZE, MAXIMUM_POOL_SIZE, KEEP_ALIVE, TimeUnit.SECONDS, sWorkQueue,sThreadFactory);//线程池是静态变量,全部的异步任务都会放到这个线程池的工做线程内执行。
一、线程池中的工做线程少于5个时,将会建立新的工做线程执行异步任务(红色表示新任务,下同)异步
二、线程池中已经有5个线程,缓冲队列未满,异步任务将会放到缓冲队列中等待ide
三、线程池中已经有5个线程,缓冲队列已满,那么线程池将新开工做线程执行异步任务工具
问题:Android的设备通常不超过2个cpu核心,过多的线程会形成线程间切换频繁,消耗系统资源。性能
四、线程池中已经有128个线程,缓冲队列已满,若是此时向线程提交任务,将会抛出RejectedExecutionException
问题:抛出的错误不catch的话会致使程序FC。
好吧,理论分析以后仍是要结合实际例子,咱们经过实现一个模拟异步获取网络图片的例子,看看会不会出现上面提到的问题。
例子:使用GridView模拟异步加载大量图片
ActivityA.java
package com.zhuozhuo; import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.ListIterator; import java.util.Map; import android.app.Activity; import android.app.AlertDialog; import android.app.Dialog; import android.app.ListActivity; import android.app.ProgressDialog; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.database.Cursor; import android.graphics.Bitmap; import android.os.AsyncTask; import android.os.Bundle; import android.provider.ContactsContract; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.AbsListView; import android.widget.AbsListView.OnScrollListener; import android.widget.Adapter; import android.widget.AdapterView; import android.widget.AdapterView.OnItemClickListener; import android.widget.BaseAdapter; import android.widget.GridView; import android.widget.ImageView; import android.widget.ListAdapter; import android.widget.SimpleAdapter; import android.widget.TextView; import android.widget.Toast; public class ActivityA extends Activity { private GridView mGridView; private List<HashMap<String, Object>> mData; private BaseAdapter mAdapter; private ProgressDialog mProgressDialog; private static final int DIALOG_PROGRESS = 0; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mGridView = (GridView) findViewById(R.id.gridview); mData = new ArrayList<HashMap<String,Object>>(); mAdapter = new CustomAdapter(); mGridView.setAdapter(mAdapter); } protected void onStart () { super.onStart(); new GetGridDataTask().execute(null);//执行获取数据的任务 } @Override protected Dialog onCreateDialog(int id) { switch (id) { case DIALOG_PROGRESS: mProgressDialog = new ProgressDialog(ActivityA.this); mProgressDialog.setMessage("正在获取数据"); mProgressDialog.setProgressStyle(ProgressDialog.STYLE_SPINNER); return mProgressDialog; } return null; } class CustomAdapter extends BaseAdapter { CustomAdapter() { } @Override public int getCount() { return mData.size(); } @Override public Object getItem(int position) { return mData.get(position); } @Override public long getItemId(int position) { return 0; } @Override public View getView(int position, View convertView, ViewGroup parent) { View view = convertView; ViewHolder vh; if(view == null) { view = LayoutInflater.from(ActivityA.this).inflate(R.layout.list_item, null); vh = new ViewHolder(); vh.tv = (TextView) view.findViewById(R.id.textView); vh.iv = (ImageView) view.findViewById(R.id.imageView); view.setTag(vh); } vh = (ViewHolder) view.getTag(); vh.tv.setText((String) mData.get(position).get("title")); Integer id = (Integer) mData.get(position).get("pic"); if(id != null) { vh.iv.setImageResource(id); } else { vh.iv.setImageBitmap(null); } FifoAsyncTask task = (FifoAsyncTask) mData.get(position).get("task"); if(task == null || task.isCancelled()) { Log.d("Test", "" + position); mData.get(position).put("task", new GetItemImageTask(position).execute(null));//执行获取图片的任务 } return view; } } static class ViewHolder { TextView tv; ImageView iv; } class GetGridDataTask extends FifoAsyncTask<Void, Void, Void> { protected void onPreExecute () { mData.clear(); mAdapter.notifyDataSetChanged(); showDialog(DIALOG_PROGRESS);//打开等待对话框 } @Override protected Void doInBackground(Void... params) { try { Thread.sleep(500);//模拟耗时的网络操做 } catch (InterruptedException e) { e.printStackTrace(); } for(int i = 0; i < 200; i++) { HashMap<String, Object> hm = new HashMap<String, Object>(); hm.put("title", "Title"); mData.add(hm); } return null; } protected void onPostExecute (Void result) { mAdapter.notifyDataSetChanged();//通知ui界面更新 dismissDialog(DIALOG_PROGRESS);//关闭等待对话框 } } class GetItemImageTask extends FifoAsyncTask<Void, Void, Void> { int pos; GetItemImageTask(int pos) { this.pos = pos; } @Override protected Void doInBackground(Void... params) { try { Thread.sleep(2000); //模拟耗时的网络操做 } catch (InterruptedException e) { e.printStackTrace(); } mData.get(pos).put("pic", R.drawable.icon); return null; } protected void onPostExecute (Void result) { mAdapter.notifyDataSetChanged();//通知ui界面更新 } } }
由运行图可见
当网络状况较差,异步任务不能尽快完成执行的状况下,新开的线程会形成listview滑动不流畅。当开启的工做线程过多时,还有出现FC的可能。
至此,你还相信万能的AsyncTask吗?至于你信不信,反正我不信。
总结:
AsyncTask可能存在新开大量线程消耗系统资源和致使应用FC的风险,所以,咱们须要根据本身的需求自定义不一样的线程池,因为篇幅问题,将留到下篇再讲。