AsyncTask解惑

来源:互联网 发布:2016最新人口普查数据 编辑:程序博客网 时间:2024/06/06 11:41
AsyncTask作为执行后台操作常见的一种实现方式,还是有必要阅读下源码,了解其实现机制的。这里是使用Android4.4的源码

使用AsyncTask的规则(这几点sdk文档都有说明)
1. AsyncTask的类必须在UI线程加载(从4.1开始系统会帮我们自动完成)
2. AsyncTask对象必须在UI线程创建
3. execute方法必须在UI线程调用
4. 不要在你的程序中去直接调用onPreExecute(), onPostExecute, doInBackground, onProgressUpdate方法
5. 一个AsyncTask对象只能执行一次,即只能调用一次execute方法,否则会报运行时异常
6. AsyncTask不是被设计为处理耗时操作的,耗时上限为几秒钟,如果要做长耗时操作,强烈建议你使用Executor,ThreadPoolExecutor以及FutureTask
7. 在1.6之前,AsyncTask是串行执行任务的,1.6的时候AsyncTask开始采用线程池里处理并行任务,但是从3.0开始,为了避免AsyncTask所带来的并发错误,AsyncTask又采用一个线程来串行执行任务

几个问题
1. AsyncTask为什么要在UI线程中创建和执行
-----AsyncTask的实现原理也是通过worker thread和Handler来实现的,工作线程在汇报工作进度及结果时和UI线程通讯是采用Handler来做的,而这个Handler只能是关联UI线程的Handler

2. AsyncTask为什么不能重用,只能执行一次execute
----这是个规则,AsyncTask源码中有对这个进行判断。

3. 为什么只能执行几秒钟的任务,不能执行耗时的操作。
stackoverflow上有人提出了同样的问题,但是我觉得答案有待商榷,  
被选中的答案大概意思是
a. AsyncTask长时间执行任务时,Activity被销毁时,后台任务仍在进行,这样会出现崩溃
b. AsyncTask常被作为内部类使用,当AsyncTask长时间执行时,其外部类对象不能被销毁,易出现内存泄漏
-----我对这个答案有所保留,我们使用WeakRefence或者worker fragment足以解决这些问题。
我认为的原因之一是串行执行的任务如果执行时间过长,其他task就不能及时进行。这样就会让app看上去反应很迟钝。
欢迎补充

4. 这里的串行执行是什么个概念?
----看下AsyncTask代码就知道了
[java] view plaincopy
  1. public static final Executor SERIAL_EXECUTOR = new SerialExecutor();  
  2. private static volatile Executor sDefaultExecutor = SERIAL_EXECUTOR;  
  3. // 执行后台任务的入口函数  
  4. // sDefaultExecutor是类的静态成员,负责管理同一个进程中所有的AsyncTask串行执行的所有子类对象  
  5. // 所以说,不管是ATask还是BTask,一次只有一个task被执行。  
  6. public final AsyncTask<Params, Progress, Result> execute(Params... params) {  
  7.     return executeOnExecutor(sDefaultExecutor, params);  
  8. }  
  9. private static class SerialExecutor implements Executor {  
  10.     final ArrayDeque<Runnable> mTasks = new ArrayDeque<Runnable>();  
  11.     Runnable mActive;  
  12.     public synchronized void execute(final Runnable r) {  
  13.         mTasks.offer(new Runnable() {  
  14.             public void run() {  
  15.                 try {  
  16.                     r.run();  
  17.                 } finally {  
  18.                     scheduleNext();  
  19.                 }  
  20.             }  
  21.         });  
  22.         if (mActive == null) {  
  23.             scheduleNext();  
  24.         }  
  25.     }  
  26.     protected synchronized void scheduleNext() {  
  27.         if ((mActive = mTasks.poll()) != null) {  
  28.             THREAD_POOL_EXECUTOR.execute(mActive);  
  29.         }  
  30.     }  
  31. }  

5.取消机制是怎么实现的
[java] view plaincopy
  1. private final AtomicBoolean mCancelled = new AtomicBoolean();  
  2. private final FutureTask<Result> mFuture;  
  3. public final boolean cancel(boolean mayInterruptIfRunning) {  
  4.     // 设置标志位  
  5.     mCancelled.set(true);  
  6.   
  7.     // 主要的实现是通过FutureTask来实现的  
  8.     return mFuture.cancel(mayInterruptIfRunning);  
  9. }  

FutureTask来实现的,FutureTask实现了Future接口,这个可以查看博客园的文章  http://www.cnblogs.com/dolphin0520/p/3949310.html

6. cancel有个参数mayInterruptIfRunning,怎么解读
------java程序员估计都清楚这个参数的意义,但作为C++出身的我碰到这个还是多少有点疑惑的。工作线程是可能执行一些阻塞的操作的,比如sleep、文件io,mayInterruptIfRunning表示是否中断这些阻塞操作,让工作线程继续往下执行。
0 0
原创粉丝点击