前言

  我们知道Android的UI主线程主要负责处理用户的按键事件、用户的触屏事件以及屏幕绘图事件等;既然UI老人家都这么忙了,我们这些开发者肯定不能不识趣的去添乱阻塞UI线程什么的,否则UI界面万一停止响应了呢——这不是招骂的节奏么?!所以我们知道用Handler+Thread的方法,在子线程中处理耗时的任务,任务完成后通过Handler通知UI主线程更新UI界面,皆大欢喜有木有。

  可是这样,还是有某些人觉得用Handler+Thread的代码会比较繁琐,当然这个某些人里面包括我们伟大的谷歌。所以AsyncTask(异步任务)在Android 1.5中横空出世;相对于Handler来说,由于比较好的封装,AsyncTask显得更加轻量级一点,适用于简单的异步处理;当然使用起来也比较简洁,果然是谷歌的亲儿子!

概述

  AsyncTask是一个抽象类,通常是被继承的命。AsyncTask的内部会维持一个静态的线程池,每个后台任务自然也会被提交到线程池中运行,同时也使用Handler+Thread的机制来调用AsyncTask的各个回调方法;回调方法是在主线程运行的,所以该干什么我们都懂(~ o ~)~zZ(赶紧跟UI界面套近乎呀)。

我们知道AsyncTask<Params, Progress, Result>是抽象类,我们可以在这里面看出它支持三种泛型:

1、Params:我们的AsyncTask要开始干活时,我们给他的输入的参数的类型,也就是传递给后台的参数

2、Progress:AsyncTask向我们报告它干活进度的参数类型,举个例子就是下载进度的百分比

3、Result:后台执行任务完成,返回的结果的参数类型

如果某个泛型我们不需要指定,我们可以大大方方的指定Void,没事AsyncTask不会伤心滴。

当然谷歌也帮我们将AsyncTask的后台任务运行的五种状态,分别是:1、准备运行,2、正在后台运行,3、进度更新,4、完成后台任务,5、取消任务。每种状态在AsyncTask中各有相应的回调方法。

1、准备运行:onPreExecute(),在任务开启时该回调方法立即在UI线程中被调用,同时也是在执行后台耗时操作前被调用;通常该方法用于完成一些初始化工作,比如在界面上显示进度条等。

2、正在后台运行:doInBackground(Params...),该回调函数由后台线程在onPreExecute()方法执行结束后立即调用,重写该方法就是后台线程将要完成的耗时任务;由于是由后台线程调用,所以我们不能直接在这里更新UI界面,应该使用publishProgress(Progress...)触发回调方法onProgressUpdate(Progress...)进行进度更新;任务计算的结果必须由该函数返回,并被传递到onPostExecute()中。

3、进度更新:onProgressUpdate(Progress...),在doInBackground()中调用publishProgress()方法更新任务的执行进度,将会在主线程中触发该方法,一般用于动态地显示一个进度条。

4、完成后台任务:onPostExecute(Result),当doInBackground()完成后,系统会自动调用onPostExecute()方法,并将doInBackground()的返回值传递给该方法。

5、取消任务:onCancelled (),在调用AsyncTask的cancel()方法时调用。

案例

参考代码:

public class MainActivity extends ActionBarActivity implements OnClickListener{

    private Button startdownload;
private ProgressBar probar;
private TextView tv; private DownTask task; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); startdownload = (Button) findViewById(R.id.startdownload);
probar = (ProgressBar) findViewById(R.id.probar);
tv = (TextView) findViewById(R.id.tv); startdownload.setOnClickListener(this); } @Override
public void onClick(View v) {
task = new DownTask();//同一个AsyncTask的execute只能调用一次
task.execute("输入参数,可为空");//调用execute后将会回调onPreExecute方法
} class DownTask extends AsyncTask<String, Integer, String>{ @Override//该方法非在主线程运行,可进行耗时操作,不可更新UI界面,其他方法为主线程运行
protected String doInBackground(String... params) {//params为execute输入的参数
for(int i = 1; i <= 100; i++){
try {//模拟下载操作
Thread.sleep(333);
publishProgress(i);//传递参数i并触发onProgressUpdate回调方法
} catch (InterruptedException e) {
e.printStackTrace();
}
}
String result = "任务已完成";
return result;//将调用onPostExecute,并将result传给该回调方法
} @Override
protected void onPreExecute() {//该回调方法执行完毕后,将会调用doInBackground
probar.setMax(100);
probar.setProgress(0);
tv.setText("开始下载");
} @Override
protected void onPostExecute(String result) {//doInBackground结束后回调该方法,结束。
Toast.makeText(MainActivity.this, result, Toast.LENGTH_SHORT).show();
tv.setText("下载完成");
} @Override
protected void onProgressUpdate(Integer... values) {//通知UI界面更新
probar.setProgress(values[0]);
} } }

布局文件:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" > <Button
android:id="@+id/startdownload"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="开始下载"/>
<ProgressBar
android:id="@+id/probar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
style="@android:style/Widget.ProgressBar.Horizontal"
/>
<TextView
android:id="@+id/tv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#000000"
android:textSize="20sp"
/> </LinearLayout>

代码讲解:

1、点击Button后先实例化一个AsyncTask的继承子类,此时将会创建一个task。接下来变执行execute(params)方法启动异步任务。(同一个AsyncTask的实例只能执行execute一次,多次执行会抛出错误)。

2、在execute()被执行后,将会触发onPreExecute()回调方法,设置进度条的初始属性。在onPreExecute()执行完毕后,将会在后台线程开始执行doInBackground(params),该方法接收execute传入的参数,进行耗时操作,这里是模拟网络文件下载任务。

3、doInBackground()在后台线程运行中,如果需要与UI主线程交互更新进度,可以调用publishProgress(values)方法,将会触发位于UI主线程运行的onProgressUpdate(values)的回调方法,代码中在这里更新进度条的进度。

4、 当后台任务执行完成后,调用onPostExecute(Result),传入的参数是doInBackground()中返回的对象。

注意事项

1、不要在同一个AsyncTask实例中多次执行execute(),正确的方法是new一个AsyncTask执行一次execute()。

2、耗时任务一定要在doInBackground()中处理,不要在其他回调方法中处理耗时任务以免引起UI主线程的阻塞。

3、不要再doInBackground()中更新UI界面,应该通过publishProgress()调用回调方法更新UI。

4、onCancelled()只能触发AsyncTask的cancel()方法,并无法取消正在线程池运行的线程任务,但可以通过标志位来停止线程任务。

5、在不同的android版本中,AsyncTask多任务运行,有些是可以并行有些则是顺序执行,不过在高版本Android中,可以通过指定参数设置线程池执行规则。

6、AsyncTask适合处理短时间的操作,长时间的操作,比如下载一个很大的视频,这就需要你使用自己的线程来下载,不管是断点下载还是其它的。

第十六章:异步处理之AsyncTask的应用的更多相关文章

  1. 《Linux命令行与shell脚本编程大全》 第十六章 学习笔记

    第十六章:创建函数 基本的脚本函数 创建函数 1.用function关键字,后面跟函数名 function name { commands } 2.函数名后面跟空圆括号,标明正在定义一个函数 name ...

  2. Gradle 1.12 翻译——第十六章. 使用文件

    有关其它已翻译的章节请关注Github上的项目:https://github.com/msdx/gradledoc/tree/1.12,或訪问:http://gradledoc.qiniudn.com ...

  3. 第十六章——处理锁、阻塞和死锁(3)——使用SQLServer Profiler侦测死锁

    原文:第十六章--处理锁.阻塞和死锁(3)--使用SQLServer Profiler侦测死锁 前言: 作为DBA,可能经常会遇到有同事或者客户反映经常发生死锁,影响了系统的使用.此时,你需要尽快侦测 ...

  4. CSS3秘笈复习:十三章&十四章&十五章&十六章&十七章

    第十三章 1.在使用浮动时,源代码的顺序非常重要.浮动元素的HTML必须处在要包围它的元素的HTML之前. 2.清楚浮动: (1).在外围div的底部添加一个清除元素:clear属性可以防止元素包围浮 ...

  5. Gradle 1.12用户指南翻译——第二十六章. War 插件

    其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Github上的地址: https://g ...

  6. Gradle 1.12用户指南翻译——第三十六章. Sonar Runner 插件

    本文由CSDN博客万一博主翻译,其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Githu ...

  7. 《HTTP 权威指南》笔记:第十六章&第十七章 国际化、内容协商与转码

    <HTTP 权威指南>笔记:第十六章 国际化 客户端通过在请求报文中的 Accept-Language 首部和 Accept-Charset 首部来告知服务器:“我理解这些语言.”服务器通 ...

  8. “全栈2019”Java多线程第二十六章:同步方法生产者与消费者线程

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  9. 读《构建之法》一、二、十六章随笔a

    第一章    概论 “软件团队要从需求分析开始,把合适的需求梳理出来,然后逐步开展后续工作”:——p3 问题:好的用户体验要从软件分析开始,那么软件分析仅仅是从用户的需求出发吗? 我的看法:需求分析是 ...

随机推荐

  1. Fiddler抓包测试App接口

    Fiddler抓包测试App接口 使用Fiddler对手机App应用进行抓包,可以对App接口进行测试,也可以了解App传输中流量使用及请求响应情况,从而测试数据传输过程中流量使用的是否合理. 抓包过 ...

  2. nginx环境下配置nagios-关于commands.cfg

    -w $ARG1$ -c $ARG2$ -M -b% -c % -f% -c % -f% -c % -f #  define command{         command_name    chec ...

  3. Linux socket 获得本地IP和广播地址

    #include <stdio.h> #include <stdlib.h> #include <sys/socket.h> #include <netine ...

  4. SVM1 线性SVM

    一.Linear Support Vector Machine 接下来的讨论假设数据都是线性可分的. 1.1 SVM的引入:增大对测量误差的容忍度 假设有训练数据和分类曲线如下图所示: 很明显,三个分 ...

  5. Swift3.0基础语法学习<四>

    协议和扩展: // // ViewController4.swift // SwiftBasicDemo // // Created by 思 彭 on 16/11/16. // Copyright ...

  6. javascript 中break、 continue、函数不能重载

    在javascript中,break与continue有着显著的差别. 如果遇到break语句,会终止最内层循环,无论后面还有多少计算. 如果遇到continue,只会终止此次循环,后面的自循环依然执 ...

  7. ELK日志管理

    ELK一般由三部分组成:logstash(日志格式化) + elasticsearch(检索) + Kibana(前台报表展示) 官网地址:https://www.elastic.co/ 本人在这用的 ...

  8. 将Discuz.net 集成到asp.net

    1,将bbs文件夹拷贝到网站的根目录下: 2,将bbs里面的bin目录,config目录,cache目录,DNT.config文件拷贝到网站根目录下: 3,修改网站web.config文件,在http ...

  9. 在updatepanel使用colorbox无效

    今天在给一个使用colorbox的页面加了一个updatepanel后,colorbox无效了,原因是以前的页面初始化colorbox是用 $(document).ready(function(){ ...

  10. Azure china服务状态报告查看网址

    https://www.azure.cn/support/service-dashboard/