AsyncTask只要是学习Android的如果你不会用这个,或者对它里面的哪个方法不熟悉那么你要仔细了解一下了,因为这个可能在开发中很长用,并且也很有用。
AsyncTask类必须在主线程中加载,在Android4.1以及以上版本中已经被系统自动加载完成。
其它不说了,直接上干货。
AsyncTask的常用方法:
1、onPreExecute(), 在主线程中执行,在异步任务开始之前执行(即一旦执行)
2、doInBackground(Params...params)
3、onProgressUpdate(Progress...values)
4、onPostExecute(Result result)
AsyncTask的使用限制:
1、AsyncTask必须在主线程中创建
2、execute()方法必须在UI线程中调用
3、不要直接在程序中调用它的上面4个常用方法
4、一个AsyncTask对象只能执行一次,否则报异常
为什么对象只能执行一次呢?看源码如下图1:
看红色部分,一旦你执行了AsyncTask对象的execute()方法,那么会先判断mStatus的状态,而mStatus的默认状态是
执行了execute()方法后,mStatus就变为Status.RUNNING,如果你多次调用execute()那么就会走execute的switch语句,抛出IllegalStateException异常。
onPreExecute()在主线程执行,看图1就会发现为什么了。
走到现在,又发现几个不知道什么鬼的东西,分别是mWorker,mWorker.mParams ,sExecutor, mFuture,这些都是什么鬼啊???
别急咱们一个一个跟它,首先来看mWorker,看它的定义如图3。
使用SourceInsight点击mWorker出现图3,知道mWorker是一个WorkerRunnable对象,顾名思义,WorkerRunnable应该是一个实现了Runnable接口的类,是不是,咱们看它的源码,看下图4-WorkerRunnable
我去,mWorker竟然是实现的 java.util.concurrent.Callable,它本身是AsyncTask的一个私有、静态、抽象类,打脸了啊^_^ !!!
没关系,打就打吧,不过 mWorker.mParams搞明白了吧 ,接着看Callable源码如图5-Callable:
Callable是Java同步包下的内容一般和线程池,Future联合起来使用,用于非阻塞线程这里不做详细论述此处我们只讨论AsyncTask(OkHttp也是使用Future与Callable+线程池实现的),如果你对Callable不了解 http://blog.csdn.net/ghsau/article/details/7451464 看这篇博客学习一下最好。
现在还剩下sExecutor和mFuture了,他们的定义吧,如下图6:
FutureTask 实现RunnableFuture ,RunnableFuture如下图7
FutureTask 具体内容可以参考 http://blog.csdn.net/javazejian/article/details/50896505 我不觉得我比别人讲的更好,所以不废话了。
OK,目前为止execute(Params...params)方法中的所有东西我们已经知道都是什么了,现在看看它们的初始化吧,如下图9
结合图1+图9,这两段代码来看,线程池执行是会执行mFuture的run()方法,源码如下 图10:
run()方法我就不在追踪了啊, 图1的mExceture.excute(mFuture)执行图10的run()方法,进入后在调用Callable的call()方法返回结果,无异常时,调用set(result),在set方法中调用finishCompletion()->在此方法中调用done()方法完整整个流程,而AsyncTask在进行初始化时,就重写了done()方法,将结果分装到handler的Message中,如图11
下面看sHandler和AsyncTaskResult源码:
图12是AsyncTaskResult的源码,它存储AsyncTask对象以及FutureTask执行结果
在来看
message = sHandler.obtainMessage(MESSAGE_POST_RESULT, new AsyncTaskResult(AsyncTask.this, result));
message.sendToTarget();
sHandler的源码如下:
执行msg.what == MESSAGE_POST_RESULT时,执行AsyncTask的私有的finish()方法如图14
在执行finish()时,执行onPostExecute()一个完整的AsyncTask过程走完。
sHandler.obtainMessage 默认将Handler的实例对象传入了啊,具体看源码