Android AsyncTask(1)-使用方法和线程池解析

当我们在android开发中需要开N个线程的时候,很多人的做法是直接new N 个 Thread。这样做不但不利于线程管理、而且大量的线程没有得到释放将会消耗性能,而且一般的Thread是没有返回线程执行完毕的返回值的,我们在Android开发中可以通过handler来告诉主线程线程是否执行完毕了。其实,Google工程师已经为我们设计好了一套强大的异步任务管理类 - AsyncTask。那么它到底强大在何处呢?接下来就开始慢慢讲解它的强大之处。

一、AsyncTask优点:

  • 内部采用了线程池机制,可以有效的管理线程。
  • 可以自定义线程池,实现多个线程按顺序同步执行、异步并发执行。
  • 提供了回调方法,后台任务执行完毕后会返回需要的数据,拿到数据后可以直接更新UI控件

二、AsyncTask用法步骤

2.1 构建自定义AsyncTask的参数

AsyncTask是一个抽象类,通常用于被继承,继承AsyncTask需要指定三个泛型参数


  1. Params :启动任务时输入参数的类型。

     【就是传进去耗时操作任务需要用到的参数,比如联网请求String类型的URL】
    
  2. Progress:后台任务执行中返回进度值的类型

     【可用于更新ProgressBar】如果要设置进度,则第二个参数一般设置为Integer
    
  3. Result:后台执行任务完成之后返回结果的类型

     【耗时操作结束后返回的类型,比如返回Bitmap】
    

    你设置的第三个参数类型是什么,doInBackGround的返回值就是什么类型



    doInBackground返回的Bitmap最终会传递到onPostExecute(Bitmap bitmap)中作为参数
    【AsyncTask底层是把异步耗时任务得到的数据result通过handler发送到了主线程】

2.2 必须实现AsyncTask的抽象方法

执行顺序从上往下。案例代码如下面2.3节代码所示:
(1)onPreExecute:执行后台耗时操作前被调用,通常用于完成一些初始化操作
(2)doInBackGround:必须实现,异步执行后台线程将要完成的任务【该方法在子线程运行】
(3)onProgressUpdate:在doInBackGround方法中调用publishProgress方法,就可以更新
任务的执行进度
(4)onPostExecute:当doInBackGround完成后,系统会自动调用,并将doInBackGround方法返回的值传给该方法

2.3 AsyncTask的用法案例

1.实例化自定义AsyncTask类,传入一个图片url和ImagView,execute执行。

new MyAsyncTask(url, imageView).execute();

2.自定义AsyncTask类代码案例:

private class MyAsyncTask extends AsyncTask<Void, Void, Bitmap> {

    private String mUrl;
    private ImageView mImageView;

    public MyAsyncTask(String url, ImageView imageView) {
        mUrl = url;
        mImageView = imageView;
    }

    @Override
    protected Bitmap doInBackground(Void... params) {
        return getBitmapFromUrl(mUrl);
    }

    @Override
    protected void onPostExecute(Bitmap bitmap) {
        super.onPostExecute(bitmap);
        if (mUrl.equals(mImageView.getTag()))
            mImageView.setImageBitmap(bitmap);
    }
} 

三、AsyncTask线程池

AsyncTask之所以如此强大,核心功臣就是它背后的线程池。
在AsyncTask中提供有两种线程池,一个是THREAD_POOL_EXECUTOR,另一个是SERIAL_EXECUTOR。
【但是,特别注意,其实在AsyncTask中只有一个线程池THREAD_POOL_EXECUTOR,只不过SERIAL_EXECUTOR实现了线程队列,最终还是使用的THREAD_POOL_EXECUTOR】

1、THREAD_POOL_EXECUTOR 
        多个任务可以在线程池中异步并发执行。
2、SERIAL_EXECUTOR
        把多个线程按串行的方式执行,所以是同步执行的。
        也就是说,只有当一个线程执行完毕之后,才会执行下个线程。

3.1 异步、同步、并行、串行的区别

  • 异步:
    发送方发出数据后,不用等接收方发回响应,接着发送下个数据包的通讯方式。
    【比如,主main函数的代码从上往下执行,new一个Thread并在子线程中途执行了sleep 10秒钟,而主main函数后面的代码不需要等子线程sleep完10秒再执行,而是直接继续执行下面的代码。】
  • 同步:
    发送方发出数据后,需要等接收方发回响应以后才发下一个数据包的通讯方式。
    【比如,主main函数的代码从上往下执行,如果中途执行了sleep 10秒钟,则后面的代码都要等10秒后才会执行。】
  • 并行:
    也称为并发。从宏观上来理解,就是在同一时间内同时执行多个线程任务。
    【比如,同时开启10张图片下载,宏观上他们是10张图同时下载的。】
  • 串行:
    可以理解为,只有当一个线程执行完毕之后,才会执行下个线程。
    【比如,10张图片下载线程串行执行,只能是第一张下载完后,才会开始执行下一张图片下载。】

3.2 THREAD_POOL_EXECUTOR 异步线程池

ThreadPoolExecutor构造函数需要传递以下几个参数:

       /**
         * @param corePoolSize 线程池的核心运行线程数量
         * 
         * @param maximumPoolSize 线程池中可以创建的最大运行线程数量【包括核心运行线程】
         * 
         * @param keepAliveTime 当线程池线程数量超过corePoolSize时,
         * 多余的空余线程在缓冲队列的存活时间,超时后将会被移除
         * 
         * @param unit 线程池维护线程所允许的空闲时间的单位,一般设置为秒
         * 
         * @param workQueue 线程池所使用的缓冲队列,可以设置缓冲队列容纳的线程数量
         * 
         * @param threadFactory 线程工厂,用于创建线程。
         *    当一个异步任务执行execute的时候将会通过该工厂new出一个thread
         * 
         */
        public ThreadPoolExecutor(int corePoolSize,
                                  int maximumPoolSize,
                                  long keepAliveTime,
                                  TimeUnit unit,
                                  BlockingQueue<Runnable> workQueue,
                                  ThreadFactory threadFactory) {
            this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,
                 threadFactory, defaultHandler);
        }

那么我们该如何理解这几个参数的意思呢?
首先我们传入线程池参数:

Executor executor = new ThreadPoolExecutor(3,5,10,
                TimeUnit.SECONDS,new LinkedBlockingDeque<Runnable>(5));

然后,同时execute执行10个任务
看图示解析:


  • 第一个参数传入是3,表示初始化核心并发运行的有三个线程,当execute任务大于3时,将会放入缓冲队列中
  • 当缓冲队列满了之后,【缓冲队列最多存5个线程】,此时已经加入进来了8个进程。
  • 但是,这还并没有结束,我们设置了第二个参数为5,就意味着,在线程池中可以创建的最大运行线程数量为5个。现在已经有3个核心线程创建了,
    所以还可以再创建两个线程。整个线程池最大支持容纳10个线程
  • 重点来了:这个时候,将会并发执行5个核心线程,是哪5个呢?答案是:1,2,3,9,10。因为新创建的两个运行线程也会开始工作。
    【1,2,3线程是核心运行线程,4,5,6,7,8在缓冲队列,9,10在运行线程池中。】
  • 那么5个线程结束之后?是继续并发执行3个,还是5个呢?答案是5个,因为运行线程已经创建为5个了。
  • 特别注意:【只有当缓冲队列满的时候,才会创建新的运行线程,否则默认只按corePoolSize的数量执行。直到超出maximumPoolSize大小,抛出异常】
  • 当再加一个线程进来时,AsyncTask中有一个拒绝策略的handler抛出异常,此时线程池已经无法再容纳更多线程任务了。
    不多说,上演示图:


从AsyncTask默认构造的THREAD_POOL_EXECUTOR可以看出,AsyncTask最大支持的缓冲任务队列是128个。


3.3 SERIAL_EXECUTOR 串行线程池

在ActivityThread中的有一段代码,设置了当Android 版本api 小于12,也就是版本小于3.1时,默认是使用AsyncTask默认的异步线程池THREAD_POOL_EXECUTOR
Android 3.1 之后使用的是 SERIAL_EXECUTOR

四、执行AsyncTask的两个方法:

1、execute 
    第一个执行方法,使用的是AsyncTask默认的线程池
2、executeOnExecutor
    第二个执行方法,需要传递进去一个Executor,可以实现自定义线程池

4.1 execute 【默认线程池执行】

刚刚提到了,AsyncTask在3.1版本之前默认使用THREAD_POOL_EXECUTOR线程池,也就是说,3.1版本之前的使用AsynTask,将会是异步并发执行

4.1.1 api8系统执行execute

我们来看2.3系统同时execute执行10个异步任务,得到的结果:
3.1版本之前的系统默认最大并发执行5个线程,缓冲线程队列最大128个。虽然开了10个异步任务,但是只能同时并发执行5个,其他的任务都得等前面5个执行完后才继续执行,接着也是5次并发执行。


4.1.2 api11以上系统执行execute

那么,3.1版本之后系统,默认是使用SERIAL_EXECUTOR串行任务执行,可以预料到异步任务将会是一个个顺序执行。

果不其然,5.0系统同时execute执行10个异步任务,是一个个线程按加入顺序同步执行的。也就是说,线程池中只有一个核心线程在工作,其他线程都要等之前的线程执行完才能执行。


4.2 executeOnExecutor 【自定义线程池执行,仅支持3.1以上系统】

很显然,AsyncTask的默认线程池完全不能满足我们的需求,这个时候就需要用到executeOnExecutor自定义线程池了。

但是,现在问题来了,3.1版本以下的系统是不支持自定义线程池执行的!所以3.1版本以下的系统需要实现同步执行异步任务,只能用Thread自己定义线程池。


4.2.1 使用默认提供的AsyncTask.THREAD_POOL_EXECUTOR线程池

AsyncTask.THREAD_POOL_EXECUTOR,代码如下:

new MyAsyncTask(progressBar).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);

这样,3.1版本以上的系统便可以实现异步并发执行任务了。我们可以看到,和3.1版本之前执行默认execute一样,采用了AsyncTask默认的线程池,最大并发执行5个线程,后面的线程都只能等之前5个结束之后再执行。


4.2.1 自定义线程池

重磅炸弹来了,我们可以自己定义线程池来执行异步并发,这样我们就可以自由控制线程池了。
下面代码可以实现10个核心任务并发执行,而且缓存队列最多能够存储100个任务,当队列满了之后还可以创建40个运行线程,瞬间爆炸。

Executor executor = new ThreadPoolExecutor(10,50,10,
                TimeUnit.SECONDS,new LinkedBlockingDeque<Runnable>(100));

new MyAsyncTask(progressBar).executeOnExecutor(executor);
new MyAsyncTask(progressBar2).executeOnExecutor(executor);

如图所示:10个线程并发执行了


五、总结

前面已经提到过了AsyncTask的优点,我们在开发中如果合理运用AsyncTask,将会比自己new Thread好很多。那么,就来总结一下AsyncTask的最佳适用场景:

  • AsyncTask比较适合有大量线程执行的情况,让线程池去管理会更加高效。
  • 假如想让几个线程按顺序执行时,可以使用AsyncTask。(前提是系统版本不能小于11)
  • 尽量使用自定义线程池,按需调节线程池参数。

5.1 AsyncTask缺点

  • 之前已经解析过了,AsyncTask在3.1系统以下默认使用AsyncTask的线程池,不可以自定义线程,假如在线程不多的情况下,是很耗性能的。因为假如你只有一个异步任务,还是会创建另外4个核心线程。

  • 容易造成内存泄露,如果持有context引用的话


线程太多有风险,开的时候需谨慎呐!下一节,将带来AsyncTask的源码解析部分。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容