标签:class java http tar com get
AsyncTask 异步任务,可以很方便的在应用中执行下载等可能阻塞UI Thread的任务,现在分析一下它的源码。
首先列出AsyncTask的一些核心方法和域:
public abstract
class AsyncTask<Params, Progress, Result> {
private static final int
CORE_POOL_SIZE = 5;
//核心线程数
private static final int MAXIMUM_POOL_SIZE = 128;
//最大线程数
private static final int KEEP_ALIVE =
1;
//超时时间,当线程数超过核心线程数时,超过这个时间的空线程就会被销毁,直到线程数等于核心线程
private static final BlockingQueue<Runnable> sPoolWorkQueue
=
new
LinkedBlockingQueue<Runnable>(10);
//用于传输和保持提交的任务。可以使用此队列与池大小进行交互
public static final Executor THREAD_POOL_EXECUTOR =
new
ThreadPoolExecutor(CORE_POOL_SIZE, MAXIMUM_POOL_SIZE,
KEEP_ALIVE,
TimeUnit.SECONDS, sPoolWorkQueue, sThreadFactory);
public static final Executor SERIAL_EXECUTOR = new SerialExecutor();
}
其实AsyncTask的的核心就是一个ThreadPoolExecutor ,这是一个Java的线程池,在生成AsyncTask的时候,从线程池取出一个线程来运行你的代码。
管理规则是这样的,
? 如果运行的线程少于
corePoolSize,则 Executor 始终首选添加新的线程,而不进行排队。
? 如果运行的线程等于或多于 corePoolSize,则
Executor 始终首选将请求加入队列,而不添加新的线程。
? 如果无法将请求加入队列,则创建新的线程,除非创建此线程超出
maximumPoolSize,在这种情况下,任务将被拒绝。
SERIAL_EXECUTOR
是维护线程安全,将新建的任务一个一个的加入到ThreadPoolExecutor 中。
通过源码解决的一些问题:
1.
API中说AsyncTask只能运行在UI
Thread,是这样么?
不是的,因为在生成AsyncTask和全局的线程池时,并没有对线程进行限制,只要所在的线程存在Looper(也就是调用过Looper.prepare的线程)都可以构造AsyncTask,所以在Service,Receiver中都可以构造AsyncTask。
2.
AsyncTask建立的任务,是被立即执行么?
不是立即执行,根据上述的规则,当目前已经有五个任务执行的时候,此时线程数等于corePoolSize,那么再构造的AsyncTask就会进入sPoolWorkQueue,直到sPoolWorkQueue满为止,这些线程都是被阻塞的,必须要有核心线程执行完成,他们才会执行。
有趣的是,如果sPoolWorkQueue满了,这时再进来任务,就是构造新线程,执行此任务(而不是队列中的任务),所以你建立的AsyncTask不是按照构造的顺序来执行的,很可能后构造的反而先执行了。
3.
AsyncTask构造的最大数量?
默认状态,在每个进程中,可以最够同时构造138个,其中同时运行128个,10个在阻塞队列之中,如果在构造就会抛出异常。
4.
如何对AsyncTask进行优化?
THREAD_POOL_EXECUTOR 是 public static
final的,所以你可以访问到这个线程池,从而动态的设定 核心线程数、最大线程数等。除非你有特殊的情况处理,否则是没有必要进行修改的。
标签:class java http tar com get
原文地址:http://www.cnblogs.com/kobe8/p/3784092.html