Android中可以扮演线程的角色还有很多,比如AsyncTask和IntentService,同时HandlerThread也是一种特殊的线程,尽管AsyncTask,IntentService以及HandlerThread的表现形式都有别于传统的线程,但是他们的本质仍然是传统的线程,对于AsyncTask来说,他的底层用到了线程池,对于IntentService和HandlerThread来说,他们的底层直接使用了线程
不同的形式的线程虽然都是线程,但是他们仍然具有不同的特性和使用场景。AsyncTask封装了线程池和Handler,他主要是为了方便开发者在子线程中更新UI。
HandlerThread是一种具有消息循环的线程,在他的内部可以使用Handler,IntantService是一个服务,系统对其惊醒了封装使其可以更方便的执行后台任务。IntentService内部采用了HandlerThread来执行任务,当任务执行完成之后,IntentService会自动退出。从任务执行的角度来看,IntentService的作用很像一个后台线程,但是INtentService是一种服务,他不容易被系统杀死,从而可以尽量保证任务的进行。而如果是一个后台线程,由于这个时候进程中没有活动的四大组件,那么这个进程的优先级就会非常低。会很容易被系统杀死,这就是IntentService的优点。
public abstract class AsyncTask<Params,Progress,Result>AsyncTask提供了四个核心方法:
package com.zxxthreadpool; import java.net.URL; import android.os.AsyncTask; public class DownloadFilesTask extends AsyncTask<URL, Integer, Long>{ @Override protected Long doInBackground(URL... urls) { // TODO Auto-generated method stub int count=urls.length; long totalSize=0; for(int i=0;i<count;i++){ totalSize+=Downloader.downloadFile(urls[i]); publishProgress((int)((i/(float)count)*100)); if(isCancelled()) break; } return totalSize; return null; } @Override protected void onProgressUpdate(Integer... progress) { // TODO Auto-generated method stub super.onProgressUpdate(progress); setProgressPercent(progress[0]); } @Override protected void onPostExecute(Long result) { // TODO Auto-generated method stub super.onPostExecute(result); showDialog("Downloaded "+result+" bytes"); } }
new DownloadFilesTask().execute(url1,url2,url3);AsyncTask在具体使用的过程中也是有一些条件限制的,主要有以下几点:
public abstract class AsyncTask<Params, Progress, Result> { public final AsyncTask<Params, Progress, Result> execute(Params... params) { return executeOnExecutor(sDefaultExecutor, params); } public final AsyncTask<Params, Progress, Result> executeOnExecutor( Executor exec, Params... params) { if (mStatus != Status.PENDING) { switch (mStatus) { case RUNNING: throw new IllegalStateException("Cannot execute task:" + " the task is alread running."); case FINISHED: throw new IllegalStateException("Cannnot execute task:" + " the task has already been executed " + "(a task can be executed only once)"); } } mStatus=Status.RUNNING; onPreExecute(); mWorker.mParams=params; exec.execute(mFuture); return this; } }
package com.zxxthreadpool; import java.util.ArrayDeque; import java.util.concurrent.Executor; import android.os.AsyncTask.Status; public abstract class AsyncTask<Params, Progress, Result> { public final AsyncTask<Params, Progress, Result> execute(Params... params) { return executeOnExecutor(sDefaultExecutor, params); } public final AsyncTask<Params, Progress, Result> executeOnExecutor( Executor exec, Params... params) { if (mStatus != Status.PENDING) { switch (mStatus) { case RUNNING: throw new IllegalStateException("Cannot execute task:" + " the task is alread running."); case FINISHED: throw new IllegalStateException("Cannnot execute task:" + " the task has already been executed " + "(a task can be executed only once)"); } } mStatus=Status.RUNNING; onPreExecute(); mWorker.mParams=params; exec.execute(mFuture); return this; } public static final Executor SERIAL_EXECUTOR=new SerialExecutor(); private static volatile Executor sDefaultExecutor=SERIAL_EXECUTOR; private static class SerialExecutor implements Executor{ final ArrayDeque<Runnable> mTasks=new ArrayDeque<Runnable>(); Runnable mActive; @Override public synchronized void execute(final Runnable r) { // TODO Auto-generated method stub mTasks.offer(new Runnable() { @Override public void run() { // TODO Auto-generated method stub try { r.run(); } finally{ scheduleNext(); } } }); if(mActive==null){ scheduleNext(); } } protected void scheduleNext() { // TODO Auto-generated method stub if((mActive=mTasks.poll())!=null){ THREAD_POOL_EXECUTOR.execute(mActive); } } } }从SerialExecutor的实现可以分析AsyncTask的排队执行的过程。首先系统会吧AsyncTask的Params参数封装为FutureTask对象,FutureTask是一个并发类,在这里他充当了Runnable的作用,接着这个FutureTask会交给SerialExecutor的execute方法处理,SerialExecutor的execute方法首先会把FutrueTask对象插入到任务队列mTasks中,如果这个时候没有正在活动的AsyncTask任务,那么就会调用SerialExecutor的scheduleNext方法来执行下一个AsyncTask任务,同时当一个AsyncTask任务执行完后AsyncTask会继续执行其他任务知道所有任务都被执行位置,这一点可以看出,默认情况下,AsyncTask是串行执行的。
mWorker=new WorkerRunnable<Params,Rsult>(){ public Result call() throws Exception{ mTaskInvoked.set(true); Process.setThreadPriority(Process.THREAD_PRORITY_BACKGROUND); return postResult(doInBackground(mParams)); } };
private Result postResult(Result result) { Message message = sHandler.obtainMessage(MESSAGE_POST_RESULT, new AsyncTask<Result>(this, result)); message.sendToTarget(); return result; }在上面代码中,postResult方法会通过sHandler发送一个MESSAGE_POST_RESULT的消息,这个sHandler的定义如下:
private static final InternalHandler sHandler=new InternalHandler(); private static class InternalHandler extends Handler{ @Override public void handleMessage(Message msg) { // TODO Auto-generated method stub super.handleMessage(msg); AsyncTaskResult result=(AsyncTaskResult)msg.obj; switch (msg.what) { case MESSAGE_POST_RESULT: result.mTask.finish(result.mData[0]); break; case MESSAGE_POST_PROGESS: result.mTask.onProgressUpdate(result.mData); break; } } }
public void finish(Result result){ if(isCancelled()){ onCancelled(result); }else{ onPostExecute(result); } mStatus=Status.FINISHED; }