番外tips: 特别喜欢一句话。假设你想了解一个人。那你从他出生開始了解吧。相同的道理,想要了解AsyncTask,就从它的源代码開始吧。

进入主题前,在说一下,开发中已经非常少使用AsyncTask了。如今流行的网络框架性能和使用都比AsyncTask好。但通过面试中遇到的一些老程序猿喜欢问这个问题,所以以下開始去分析。

public abstract class AsyncTask<Params, Progress, Result>

从声明来看,AsyncTask是一个抽象泛型类。我们都知道。我们创建AsyncTask的时候,经常处理几个方法

onPreExecute() //此方法在在主线程运行。用于后台任务进行前做一些准备工作
doInBackground(Params... params) //此方法在子线程运行,用来处理后台任务。像网络请求。耗时处理等操作
onProgressUpdate(Progress... values) //此方法在在主线程运,在doInBackground通过publishProgress来调用,用来更新进度
onPostExecute(Result result) //此方法在在主线程运行,后台任务处理完成调用。并返回后台任务的结果

回到AsyncTask上,在这个类中有这么段代码

static {
ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(
CORE_POOL_SIZE, MAXIMUM_POOL_SIZE, KEEP_ALIVE_SECONDS, TimeUnit.SECONDS,
sPoolWorkQueue, sThreadFactory);
threadPoolExecutor.allowCoreThreadTimeOut(true);
THREAD_POOL_EXECUTOR = threadPoolExecutor;
}
//--------------------------------------------------不华丽的切割线
private static class InternalHandler extends Handler {
public InternalHandler() {
super(Looper.getMainLooper());
} @SuppressWarnings({"unchecked", "RawUseOfParameterizedType"})
@Override
public void handleMessage(Message msg) {
AsyncTaskResult<? > result = (AsyncTaskResult<?>) msg.obj;
switch (msg.what) {
case MESSAGE_POST_RESULT:
// There is only one result
result.mTask.finish(result.mData[0]);
break;
case MESSAGE_POST_PROGRESS:
result.mTask.onProgressUpdate(result.mData);
break;
}
}
}

静态代码块的作用我就不赘述了,我要说的是,AsyncTask内部也是通过线程池+Handler的方式实现的,这样一说似乎大家瞬间理解了,可是,它内部是非常复杂的。我们继续往下看AsyncTask的构造方法:

 public AsyncTask() {
mWorker = new WorkerRunnable<Params, Result>() {
public Result call() throws Exception {
mTaskInvoked.set(true);
Result result = null;
try {
Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);
//noinspection unchecked
result = doInBackground(mParams);
Binder.flushPendingCommands();
} catch (Throwable tr) {
mCancelled.set(true);
throw tr;
} finally {
postResult(result);
}
return result;
}
}; mFuture = new FutureTask<Result>(mWorker) {
@Override
protected void done() {
try {
postResultIfNotInvoked(get());
} catch (InterruptedException e) {
android.util.Log.w(LOG_TAG, e);
} catch (ExecutionException e) {
throw new RuntimeException("An error occurred while executing doInBackground()",
e.getCause());
} catch (CancellationException e) {
postResultIfNotInvoked(null);
}
}
};
}

在构造方法中new了两个非常重要的对象,以下看一下两个类的声明。需说明一下,WorkerRunnable是在AsyncTask定义的一个抽象泛型类。

private static abstract class WorkerRunnable<Params, Result> implements Callable<Result> //WorkerRunnable实现了Callable接口

public class FutureTask<V> implements RunnableFuture<V>
public interface RunnableFuture<V> extends Runnable, Future<V>
//FutureTask说白了就是一个Runnable对象

回到构造方法,在new WorkerRunnable 对象的时候通过result = doInBackground(mParams);给result赋值。然后会调用postResult(result)方法。看一下这种方法

 private Result postResult(Result result) {
@SuppressWarnings("unchecked")
Message message = getHandler().obtainMessage(MESSAGE_POST_RESULT,
new AsyncTaskResult<Result>(this, result));
message.sendToTarget();
return result;
}
这种方法通过Message把result发送到Handler中,Handler终于传回onPostExecute(result)这种方法中。这里是通过前面声明Handler的时候,调用finish这种方法的,顺便看一下finish方法的源代码 private void finish(Result result) {
if (isCancelled()) {
onCancelled(result);
} else {
onPostExecute(result);
}
mStatus = Status.FINISHED;
}

,并把WorkerRunnable对象当做參数传递给FutureTask,既然传到FutureTask中了。以下就不继续看FutureTask 源代码先了。由于那是Runnable对象(里面就是进入run方法,完了然后set方法。赋值给result,最后在AsyncTask通过get()调用FutureTask 对象的get方法获取result)当然,AsyncTask是通过线程池来处理的。当我们创建完AsyncTask的时候。通过调用AsyncTask的execute方法,里面 通过exec.execute(mFuture)开启线程池去跑任务。跑完后回调FutureTask的done()方法。这种方法又 调用postResultIfNotInvoked(get())方法。这里是调用AsyncTask的get()方法从而调用FutureTask 对象的get方法获取result。

以下看看 execute方法的源代码。

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 already running.");
case FINISHED:
throw new IllegalStateException("Cannot 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;
}

通过上面的回想,AsyncTask的分析也基本完了。上述比較乱。但无非是AsyncTask的那几个经常用法何时被调用的。我再总结一下这个过程。首先是通过AsyncTask的构造方法初始化了两个对象,各自是WorkerRunnable和FutureTask,在WorkerRunnable中的call()方法通过result = doInBackground(mParams)这种方法调用doInBackground(mParams)方法,这里在说明一下,并不是在WorkerRunnable运行doInBackground方法,而是在FutureTask中,传入WorkerRunnable对象,然后通过调用AsyncTask的execute方法,把传入的FutureTask參数交给线程池去运行。

在这个execute方法中,调用executeOnExecutor方法,这种方法 中运行了onPreExecute()方法。当线程池跑完了后,回调FutureTask的done方法。done方法中调用AsyncTask的get()方法从而调用FutureTask 对象的get方法获取result并通过postResultIfNotInvoked(result)这种方法。这种方法中又调用postResult(result)方法,这种方法通过Message把result传递到Handler中。在Handler中调用onPostExecute(result)。终于把结果返回。这里另一个onProgressUpdate方法,这里在看一下源代码吧

 @WorkerThread
protected final void publishProgress(Progress... values) {
if (!isCancelled()) {
getHandler().obtainMessage(MESSAGE_POST_PROGRESS,
new AsyncTaskResult<Progress>(this, values)).sendToTarget();
}
}

在此,原理分析完成。以下这两篇文章分析了它的缺点。写得非常好。大家去阅读以下,这里仅转载过来做參考并总结,文章末尾给出链接。

1、线程池中已经有128个线程,缓冲队列已满。假设此时向线程提交任务,将会抛出RejectedExecutionException。

过多的线程会引起大量消耗系统资源和导致应用FC的风险。

2、AsyncTask不会随着Activity的销毁而销毁。直到doInBackground()方法运行完成。假设我们的Activity销毁之前。没有取消 AsyncTask。这有可能让我们的AsyncTask崩溃(crash)。由于它想要处理的view已经不存在了。所以,我们总是必须确保在销毁活动之前取消任务。

假设在doInBackgroud里有一个不可中断的操作,比方BitmapFactory.decodeStream(),调用了cancle() 也未必能真正地取消任务。关于这个问题。在4.4后的AsyncTask中,都有推断是取消的方法isCancelled(),可能參考的这些作者都分析较早的版本号。当然,这是笔者落后的原因。

3、假设AsyncTask被声明为Activity的非静态的内部类。那么AsyncTask会保留一个对创建了AsyncTask的Activity的引用。

假设Activity已经被销毁,AsyncTask的后台线程还在运行。它将继续在内存里保留这个引用,导致Activity无法被回收,引起内存泄露。

4、屏幕旋转或Activity在后台被系统杀掉等情况会导致Activity的又一次创建,之前运行的AsyncTask会持有一个之前Activity的引用,这个引用已经无效,这时调用onPostExecute()再去更新界面将不再生效。

參考博客出处

mylzc - AsyncTask的缺陷

viclee108 -AsyncTask的缺陷和问题

AsyncTask的原理和缺点的更多相关文章

  1. AJAX工作原理与缺点

    1.概念:什么是AJAXAJAX全称为“Asynchronous JavaScript and XML”(异步JavaScript和XML),是一种创建交互式网页应用的网页开发技术.2.为什么要使用他 ...

  2. 看完这篇,再也不怕被问到 AsyncTask 的原理了

    本文很多资料基于Google Developer官方对AsyncTask的最新介绍. AsyncTask 是什么 AsyncTask is designed to be a helper class ...

  3. Android AsyncTask运作原理和源码分析

    自10年大量看源码后,很少看了,抽时间把最新的源码看看! public abstract class AsyncTask<Params, Progress, Result> {     p ...

  4. Android查缺补漏(线程篇)-- AsyncTask的使用及原理详细分析

    本文作者:CodingBlock 文章链接:http://www.cnblogs.com/codingblock/p/8515304.html 一.AsyncTask的使用 AsyncTask是一种轻 ...

  5. android多线程-AsyncTask之工作原理深入解析(上)

    关联文章: Android 多线程之HandlerThread 完全详解 Android 多线程之IntentService 完全详解 android多线程-AsyncTask之工作原理深入解析(上) ...

  6. AsyncTask 解析

    [转载自 http://blog.csdn.net/yanbober ] 1 背景 Android异步处理机制一直都是Android的一个核心,也是应用工程师面试的一个知识点.前面我们分析了Handl ...

  7. Android Learning Note -- AsyncTask浅尝

    AsyncTask 实现原理 AsyncTask是Android提供的轻量级异步类,可以直接继承AsyncTask在类中实现异步操作,并提供接口反馈当前的异步执行程度(通过接口实现UI进度更新),最后 ...

  8. AOP的实现原理

    1 AOP各种的实现 AOP就是面向切面编程,我们可以从几个层面来实现AOP. 在编译器修改源代码,在运行期字节码加载前修改字节码或字节码加载后动态创建代理类的字节码,以下是各种实现机制的比较. 类别 ...

  9. AsyncTask介绍

    AsyncTask介绍 AsyncTask比Handler更轻量级一些,适用于简单的异步处理. 使用AsyncTask时,注意重写以下几个方法: 1. doInBackground() 作用:执行后台 ...

随机推荐

  1. Problem A: 深入浅出学算法022-汉诺塔问题II

    #include<stdio.h> void hanio(int n,char a,char b,char c) { ) printf("%c->%c\n",a, ...

  2. 【MySQL笔记】用户管理

    1.账户管理 1.1登录和退出MySQL服务器 MySQL –hhostname|hostIP –P port –u username –p[password] databaseName –e &qu ...

  3. 忘记原来的myql的root的密码;

    修改的用户都以root为列.一.拥有原来的myql的root的密码: 方法一:在mysql系统外,使用mysqladmin# mysqladmin -u root -p password " ...

  4. HDU 4639 Hehe (2013多校4 1008 水题)

    Hehe Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  5. inno setup 脚本常用修改 转

    http://blog.sina.com.cn/s/blog_72c2eb350100y2sa.html 有人提及想更换安装界面的图片,其实方法很简单,只需要修改inno setup安装目录下的Wiz ...

  6. 【js UUID】JS生成UUID 使用

    * 生成UUID * @returns */ function UUID() { var s = []; var hexDigits = "0123456789abcdef"; f ...

  7. [java] 模拟QPS

    在WEB服务器端,每日的访问量巨大.在非生产环境需要对服务器进行压力测试,一般使用后台线程和Sleep方式来模拟线上的压力.这里使用ScheduledExecutorService实现一种简单的QPS ...

  8. [转]SQLServer和Oracle,存储过程区别,常用函数对比

    本文转自:http://www.cnblogs.com/neru/archive/2011/08/18/2144049.html 以前一直用sqlserver,只有很少的一点oracle的经验,现在要 ...

  9. python wheel 包命名规则和 abi 兼容

    wheel 包的命名规定 wheel 包的命名格式为 {distribution}-{version}(-{build tag})?-{python tag}-{abi tag}-{platform ...

  10. centos使用密钥替换密码登录服务器

    一.首先登陆centos,切换用户,切换到你要免密码登陆的用户,进入到家目录,以下我以admin为例,命令:su admincd ~ 二.创建钥匙,命令:ssh-keygen -t rsa,一路按Y搞 ...