【Android 异步操作】AsyncTask 异步任务 ( AsyncTask 异步任务执行方法 execute 方法相关源码解析 )

简介: 【Android 异步操作】AsyncTask 异步任务 ( AsyncTask 异步任务执行方法 execute 方法相关源码解析 )

文章目录

一、AsyncTask 异步任务执行方法 execute() 引入

二、AsyncTask 异步任务执行方法 execute()

三、sDefaultExecutor 线程池解析

四、executeOnExecutor 方法解析

五、AsyncTask 异步任务执行方法 execute() 相关源码注释





一、AsyncTask 异步任务执行方法 execute() 引入


上一篇博客中 【Android 异步操作】AsyncTask 异步任务 ( 参数简介 | 方法简介 | 使用方法 | AsyncTask 源码分析 ) , 讲解了 AsyncTask<Params, Progress, Result> 异步任务的构造函数 ;


异步任务执行有两个方法 :


构造异步任务 : 调用 AsyncTask 的构造函数 , 创建 AsyncTask 异步任务对象 ;

执行异步任务 : 调用 AsyncTask 异步任务对象的 execute() 方法 , 执行异步任务 ;

本篇博客中讲解执行函数 AsyncTask<Params, Progress, Result> execute(Params… params) ;


// 1. 创建 AsyncTask 异步任务
MyAsyncTask mMyAsyncTask = new MyAsyncTask();
// 2. 执行 AsyncTask 异步任务
mMyAsyncTask.execute();





二、AsyncTask 异步任务执行方法 execute()


AsyncTask<Params, Progress, Result> execute(Params… params) 方法作用 : AsyncTask 异步任务的 execute() 函数作用就是提交任务 , 其提交的任务就是 FutureTask , 其使用的是 线程池 提交任务 ;



① 参数 : 使用指定的参数 , 执行任务 ;


② 返回值 : 这个任务返回 AsyncTask<Params, Progress, Result> 本身对象 , 以便调用者可以持有该异步任务的引用 ;


③ 任务调度 : 该方法在队列上调度一个任务 , 该任务在一个单独的后台线程 , 或线程池中执行 ; 第一次引入后 , 异步任务在单独后台线程中被串行执行 ;


④ 不同版本的执行任务载体 :


Android 1.6(API 级别 4)及以下的版本是在一个后台线程中串行执行 ;

Android 2.0(API 级别 5)及以上的版本是在线程池中串行执行 ;

Android 3.0(API 级别 11)及以上的版本有回复到在一个后台线程中串行执行 ;

在该类中 , 最终调用了 executeOnExecutor() 方法 , 使用 sDefaultExecutor 默认串行线程池执行任务 ;


public abstract class AsyncTask<Params, Progress, Result> {
    @MainThread
    public final AsyncTask<Params, Progress, Result> execute(Params... params) {
        return executeOnExecutor(sDefaultExecutor, params);
    }
}


如果同时调用 AsyncTask 异步任务的 execute() 函数执行任务 , 其任务会逐个串行执行 , 不是并行的 ;


AsyncTask<Params, Progress, Result> 对象的 execute() 执行的任务是有序的 ;






三、sDefaultExecutor 线程池解析


上述执行异步任务的方法 executeOnExecutor() , 传入了 sDefaultExecutor 线程池作为参数 , 本节分析 sDefaultExecutor 线程池 ;



sDefaultExecutor 是成员变量 , 其声明类型是 Executor , 实际类型是自定义的线程池 SerialExecutor() ;



SerialExecutor 是一个线程池 , 执行任务 , 每次只执行一个任务 , 按照串行顺序执行 ; 该序列对于特定进程来说是全局的 , 即一个进程只有一个该序列 ;



在该线程池中 , 维护了如下两个成员 :


任务队列 ArrayDeque mTasks

当前执行任务 Runnable mActive


该线程池中 void execute(final Runnable r) 执行任务方法 :


执行当前传入的任务 : 自己创建一个 Runnable , 在其 run() 方法中执行传入的参数的 r 的 run() 方法 ;

执行下一个任务 : 执行完毕当前任务后 , 调用 scheduleNext() 执行下一个任务


scheduleNext() 方法 用于执行下一个任务 , 从队列中取出一个任务 , 如果取出的任务不为空 , 那么就执行该任务 ; 调用 SerialExecutor 类的 execute(final Runnable r) 方法执行该任务 ;


public abstract class AsyncTask<Params, Progress, Result> {
    private static final String LOG_TAG = "AsyncTask";
    /**
     * 自定义线程池 ; 
     * 一个线程池 , 执行任务 , 每次只执行一个任务 , 按照串行顺序执行 ; 
     * 该序列对于特定进程来说是全局的 , 即一个进程只有一个该序列 ; 
     */
    public static final Executor SERIAL_EXECUTOR = new SerialExecutor();
    @UnsupportedAppUsage
    private static volatile Executor sDefaultExecutor = SERIAL_EXECUTOR;
    /**
     * 自定义线程池 ; 
     * 一个线程池 , 执行任务 , 每次只执行一个任务 , 按照串行顺序执行 ; 
     * 该序列对于特定进程来说是全局的 , 即一个进程只有一个该序列 ; 
     */
    private static class SerialExecutor implements Executor {
      // 任务队列 
        final ArrayDeque<Runnable> mTasks = new ArrayDeque<Runnable>();
        // 当前执行的任务 
        Runnable mActive;
  // 执行任务
        public synchronized void execute(final Runnable r) {
          // 自己创建一个 Runnable , 在其 run() 方法中执行如下方法 : 
          // 传入的 execute(final Runnable r) 方法参数的 r 的 run() 方法 ; 
            mTasks.offer(new Runnable() {
                public void run() {
                    try {
                        r.run();
                    } finally {
                      // 执行完毕当前任务后 , 执行下一个任务
                        scheduleNext();
                    }
                }
            });
    // 如果当前任务为空 , 那么执行下一个任务 
            if (mActive == null) {
                scheduleNext();
            }
        }
  /**
   * 执行下一个任务 
   * 从队列中取出一个任务 , 如果取出的任务不为空 , 那么就执行该任务 ; 
   * 调用 SerialExecutor 类的 execute(final Runnable r) 方法执行该任务 ; 
   */
        protected synchronized void scheduleNext() {
            if ((mActive = mTasks.poll()) != null) {
              // 执行队列中的任务的操作 
                THREAD_POOL_EXECUTOR.execute(mActive);
            }
        }
    }
}





四、executeOnExecutor 方法解析


executeOnExecutor 方法是异步任务执行的核心方法 ;



执行流程如下 :



① 判定合法性 : AsyncTask 异步任务只能执行一次 , 必须确保该 AsyncTask 异步任务没有执行过 , 如果执行过直接抛出异常 ;


② 设置运行状态 : 将该异步任务状态设置成 Status.RUNNING 状态 , 防止第二次被执行 ;


③ 主线程初始化 : 执行 onPreExecute() 方法 , 用户可以在该方法中初始化 UI , 该操作在 UI 主线程中运行 ;


④ 子线程后台任务执行 : 执行 FutureTask 中的 Callable 任务 , 也就是异步任务 , 该操作在子线程中运行 ;



public abstract class AsyncTask<Params, Progress, Result> { 
    /**
     * 使用指定的参数 , 执行任务 ; 
     * 这个任务返回 AsyncTask<Params, Progress, Result> 本身对象 , 
     * 一遍调用者可以持有该异步任务的引用 ; 
     * 
     * 该方法通常与 THREAD_POOL_EXECUTOR 一同使用 , 允许多个任务在一个线程池中串行执行 , 
     * 该线程池由 AsyncTask 异步任务管理 , 也可以使用自己的线程池定制相关行为 ; 
     */
    @MainThread
    public final AsyncTask<Params, Progress, Result> executeOnExecutor(Executor exec,
            Params... params) {
        // 首先判定当前状态是否合法 , 确保该 AsyncTask 异步任务没有执行过 , 
        // 每个 AsyncTask 异步任务创建后只能执行一次 ; 
        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;
  // 执行 doInBackground() 之前执行的方法 
        onPreExecute();
  // 设置 WorkerRunnable 的 Params[] mParams 参数
        mWorker.mParams = params;
        // 执行 FutureTask 中的 Callable 任务
        exec.execute(mFuture);
        return this;
    }
}






五、AsyncTask 异步任务执行方法 execute() 相关源码注释


public abstract class AsyncTask<Params, Progress, Result> {
    private static final String LOG_TAG = "AsyncTask";
    /**
     * 自定义线程池 ; 
     * 一个线程池 , 执行任务 , 每次只执行一个任务 , 按照串行顺序执行 ; 
     * 该序列对于特定进程来说是全局的 , 即一个进程只有一个该序列 ; 
     */
    public static final Executor SERIAL_EXECUTOR = new SerialExecutor();
    @UnsupportedAppUsage
    private static volatile Executor sDefaultExecutor = SERIAL_EXECUTOR;
    /**
     * 自定义线程池 ; 
     * 一个线程池 , 执行任务 , 每次只执行一个任务 , 按照串行顺序执行 ; 
     * 该序列对于特定进程来说是全局的 , 即一个进程只有一个该序列 ; 
     */
    private static class SerialExecutor implements Executor {
      // 任务队列 
        final ArrayDeque<Runnable> mTasks = new ArrayDeque<Runnable>();
        // 当前执行的任务 
        Runnable mActive;
  // 执行任务
        public synchronized void execute(final Runnable r) {
          // 自己创建一个 Runnable , 在其 run() 方法中执行如下方法 : 
          // 传入的 execute(final Runnable r) 方法参数的 r 的 run() 方法 ; 
            mTasks.offer(new Runnable() {
                public void run() {
                    try {
                        r.run();
                    } finally {
                      // 执行完毕当前任务后 , 执行下一个任务
                        scheduleNext();
                    }
                }
            });
    // 如果当前任务为空 , 那么执行下一个任务 
            if (mActive == null) {
                scheduleNext();
            }
        }
  /**
   * 执行下一个任务 
   * 从队列中取出一个任务 , 如果取出的任务不为空 , 那么就执行该任务 ; 
   * 调用 SerialExecutor 类的 execute(final Runnable r) 方法执行该任务 ; 
   */
        protected synchronized void scheduleNext() {
            if ((mActive = mTasks.poll()) != null) {
              // 执行队列中的任务的操作 
                THREAD_POOL_EXECUTOR.execute(mActive);
            }
        }
    }
    /**
     * 使用指定的参数 , 执行任务 ; 
     * 这个任务返回 AsyncTask<Params, Progress, Result> 本身对象 , 
     * 一遍调用者可以持有该异步任务的引用 ; 
     * 
     * 该方法在队列上调度一个任务 , 该任务在一个单独的后台线程  , 或线程池中执行 ; 
     * 第一次引入后 , 异步任务在单独后台线程中被串行执行 ; 
     * Android 1.6(API 级别 4)及以下的版本是在一个后台线程中串行执行 ; 
     * Android 2.0(API 级别 5)及以上的版本是在线程池中串行执行 ; 
     * Android 3.0(API 级别 11)及以上的版本有回复到在一个后台线程中串行执行 ; 
     *
     * <p>改方法必须在 UI 线程中调用 
     *
     * @param 要执行的任务的参数.
     *
     * @return 返回异步任务 AsyncTask 本身 .
     */
    @MainThread
    public final AsyncTask<Params, Progress, Result> execute(Params... params) {
        return executeOnExecutor(sDefaultExecutor, params);
    }
    /**
     * 使用指定的参数 , 执行任务 ; 
     * 这个任务返回 AsyncTask<Params, Progress, Result> 本身对象 , 
     * 一遍调用者可以持有该异步任务的引用 ; 
     * 
     * 该方法通常与 THREAD_POOL_EXECUTOR 一同使用 , 允许多个任务在一个线程池中串行执行 , 
     * 该线程池由 AsyncTask 异步任务管理 , 也可以使用自己的线程池定制相关行为 ; 
     */
    @MainThread
    public final AsyncTask<Params, Progress, Result> executeOnExecutor(Executor exec,
            Params... params) {
        // 首先判定当前状态是否合法 , 确保该 AsyncTask 异步任务没有执行过 , 
        // 每个 AsyncTask 异步任务创建后只能执行一次 ; 
        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;
  // 执行 doInBackground() 之前执行的方法 
        onPreExecute();
  // 设置 WorkerRunnable 的 Params[] mParams 参数
        mWorker.mParams = params;
        // 执行 FutureTask 中的 Callable 任务
        exec.execute(mFuture);
        return this;
    }
}




目录
相关文章
|
5月前
|
监控 安全 网络安全
深入解析PDCERF:网络安全应急响应的六阶段方法
PDCERF是网络安全应急响应的六阶段方法,涵盖准备、检测、抑制、根除、恢复和跟进。本文详细解析各阶段目标与操作步骤,并附图例,助读者理解与应用,提升组织应对安全事件的能力。
780 89
|
8月前
|
人工智能
歌词结构的巧妙安排:写歌词的方法与技巧解析,妙笔生词AI智能写歌词软件
歌词创作是一门艺术,关键在于巧妙的结构安排。开头需迅速吸引听众,主体部分要坚实且富有逻辑,结尾则应留下深刻印象。《妙笔生词智能写歌词软件》提供多种 AI 功能,帮助创作者找到灵感,优化歌词结构,写出打动人心的作品。
|
6月前
|
存储 Java 开发者
浅析JVM方法解析、创建和链接
上一篇文章《你知道Java类是如何被加载的吗?》分析了HotSpot是如何加载Java类的,本文再来分析下Hotspot又是如何解析、创建和链接类方法的。
402 132
|
4月前
|
编解码 缓存 Prometheus
「ximagine」业余爱好者的非专业显示器测试流程规范,同时也是本账号输出内容的数据来源!如何测试显示器?荒岛整理总结出多种测试方法和注意事项,以及粗浅的原理解析!
本期内容为「ximagine」频道《显示器测试流程》的规范及标准,我们主要使用Calman、DisplayCAL、i1Profiler等软件及CA410、Spyder X、i1Pro 2等设备,是我们目前制作内容数据的重要来源,我们深知所做的仍是比较表面的活儿,和工程师、科研人员相比有着不小的差距,测试并不复杂,但是相当繁琐,收集整理测试无不花费大量时间精力,内容不完善或者有错误的地方,希望大佬指出我们好改进!
294 16
「ximagine」业余爱好者的非专业显示器测试流程规范,同时也是本账号输出内容的数据来源!如何测试显示器?荒岛整理总结出多种测试方法和注意事项,以及粗浅的原理解析!
|
3月前
|
JSON 监控 网络协议
Bilibili直播信息流:连接方法与数据解析
本文详细介绍了自行实现B站直播WebSocket连接的完整流程。解析了基于WebSocket的应用层协议结构,涵盖认证包构建、心跳机制维护及数据包解析步骤,为开发者定制直播数据监控提供了完整技术方案。
|
3月前
|
安全 IDE Java
重学Java基础篇—Java Object类常用方法深度解析
Java中,Object类作为所有类的超类,提供了多个核心方法以支持对象的基本行为。其中,`toString()`用于对象的字符串表示,重写时应包含关键信息;`equals()`与`hashCode()`需成对重写,确保对象等价判断的一致性;`getClass()`用于运行时类型识别;`clone()`实现对象复制,需区分浅拷贝与深拷贝;`wait()/notify()`支持线程协作。此外,`finalize()`已过时,建议使用更安全的资源管理方式。合理运用这些方法,并遵循最佳实践,可提升代码质量与健壮性。
105 1
|
6月前
|
安全 Ubuntu Shell
深入解析 vsftpd 2.3.4 的笑脸漏洞及其检测方法
本文详细解析了 vsftpd 2.3.4 版本中的“笑脸漏洞”,该漏洞允许攻击者通过特定用户名和密码触发后门,获取远程代码执行权限。文章提供了漏洞概述、影响范围及一个 Python 脚本,用于检测目标服务器是否受此漏洞影响。通过连接至目标服务器并尝试登录特定用户名,脚本能够判断服务器是否存在该漏洞,并给出相应的警告信息。
395 84
|
3月前
|
传感器 监控 Java
Java代码结构解析:类、方法、主函数(1分钟解剖室)
### Java代码结构简介 掌握Java代码结构如同拥有程序世界的建筑蓝图,类、方法和主函数构成“黄金三角”。类是独立的容器,承载成员变量和方法;方法实现特定功能,参数控制输入环境;主函数是程序入口。常见错误包括类名与文件名不匹配、忘记static修饰符和花括号未闭合。通过实战案例学习电商系统、游戏角色控制和物联网设备监控,理解类的作用、方法类型和主函数任务,避免典型错误,逐步提升编程能力。 **脑图速记法**:类如太空站,方法即舱段;main是发射台,static不能换;文件名对仗,括号要成双;参数是坐标,void不返航。
143 5
|
8月前
|
人工智能
写歌词的技巧和方法全解析:开启你的音乐创作之旅,妙笔生词智能写歌词软件
怀揣音乐梦想,渴望用歌词抒发情感?掌握关键技巧,你也能踏上创作之旅。灵感来自生活点滴,主题明确,语言简洁,韵律和谐。借助“妙笔生词智能写歌词软件”,AI辅助创作,轻松写出动人歌词,实现音乐梦想。
|
3月前
|
消息中间件 JavaScript 前端开发
最细最有条理解析:事件循环(消息循环)是什么?为什么JS需要异步
度一教育的袁进老师谈到他的理解:单线程是异步产生的原因,事件循环是异步的实现方式。 本质是因为渲染进程因为计算机图形学的限制,只能是单线程。所以需要“异步”这个技术思想来解决页面阻塞的问题,而“事件循环”是实现“异步”这个技术思想的最主要的技术手段。 但事件循环并不是全部的技术手段,比如Promise,虽然受事件循环管理,但是如果没有事件循环,单一Promise依然能实现异步不是吗? 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您

推荐镜像

更多