热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

详解Android中AsyncTask的使用

这篇文章主要介绍了详解Android中AsyncTask的使用的相关资料,这里提供实例帮助大家学习理解这部分内容,需要的朋友可以参考下

详解Android 中AsyncTask 的使用

1、首先我们来看看AsyncTask 的介绍:  

   Handler 和 AsyncTask 都是android 中用来实现异步任务处理的方式;其中:

    Handler 实例向 UI 线程发送消息,完成界面更新, 

        优点:对整个过程控制的比较精细;
        缺点:代码相对臃肿,多个任务同时执行时,不易对线程进行精确的控制;

    AsyncTask :比Handler 更轻量级一些,适用于简单的异步处理; 

        优点:简单 | 快捷 | 过程可控;
        缺点:使用多个异步操作时就变得复杂起来;

2、AsyncTask 的定义:(AsyncTask 定义了三种泛型类型) 

  public abstract class AsyncTask{...} 

    说明: 

        Params :启动任务执行的输入参数,例如:HTTP 请求的URL;
        Progress: 后台任务执行的百分比;
        Result:后台执行任务最终返回的结果,比如String;

3、AsyncTask 异步任务的执行步骤:(以下方法除execute(Params... params),在AsyncTask中重写),下列是相关方法的介绍:

    A、execute(Params... params) : 

        执行一个异步任务,需要我们在UI线程中调用,触发任务

    B、OnPreExecute(): 

        execute(Params... params)调用后立即执行,一般用于在执行后台任务前对UI做一些标记; 例如,可以在此处显示进度对话框;

    C、doInBackground(Params.. params): 

        onPreExecute() 完成后执行,后台执行,处理比较耗时的操作;此处不能操作UI,执行的过程中调用publishProgress(Progress... values)来更新进度信息;

    D、onProgressUpdate(Progress... values): 

        在调用publicshProgress(Progress... values)方法执行,直接将进度信息更新到UI组建上;此方法在主线程上执行,用于显示任务执行的进度;

    E、onPostExecute(Result result): 

        此方法在主线程中执行,当后台的操作结束时,此方法会被调用,计算结果作为参数传递到此方法中,直接将结果显示到UI组建上。

    F、cancel(); : 

        取消一个正在执行的任务,在UI线程中完成,用AsyncTask的对象进行调用,参数为true/false;

4、使用AsyncTask 时注意事项: 

    A、异步任务实例必须在UI线程中创建;
    B、execute(Params... params) 方法必须在UI线程中调用;
    C、不要手动的调onPreExecute().doInBackground().onProgressUpdate().onPostExecute()这几个方法;
    D、不能在doInBackground(Params... params) 中更改组件信息;
    E、一个任务实例只能执行一次,如果执行第二次会抛出异常;

5、案例:使用AsyncTask 实现图片的下载:

    Activity类,主程序的入口:

  public class MainActivity extends Activity { 
 
  // 程序入口 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_main); 
    MyAsyncTask my = new MyAsyncTask(); 
    my.execute("http://photocdn.sohu.com/20110927/Img320705637.jpg"); 
  } 
 
} 

   AsyncTask 派生类,实现异步任务:

  package com.sun.asynctask; 
 
import java.io.ByteArrayOutputStream; 
import java.io.InputStream; 
import java.net.HttpURLConnection; 
import java.net.URL; 
 
import org.apache.http.HttpConnection; 
import org.apache.http.HttpEntity; 
import org.apache.http.HttpResponse; 
import org.apache.http.HttpStatus; 
import org.apache.http.client.HttpClient; 
import org.apache.http.client.methods.HttpGet; 
import org.apache.http.impl.client.DefaultHttpClient; 
 
import android.graphics.Bitmap; 
import android.graphics.BitmapFactory; 
import android.os.AsyncTask; 
import android.util.Log; 
 
/** 
 * 异步任务,实现网页内容的获取 
 * 
 * 
 * 生成该类的对象,并调用execute方法之后 
 * 
 * 首先执行的是onProExecute() 方法, 
 * 
 * 其次执行的doInBackground()方法 
 */ 
public class MyAsyncTask extends AsyncTask { 
 
  /** 
   * 在execute() 方法执行后立即执行,运行在UI线程中, 
   * 在后台任务开始前执行,用于标识UI界面 
   */ 
  protected void onPreExecute() { 
    super.onPreExecute(); 
    Log.i("msg","onPreExecute()..."); 
  } 
 
  /** 
   * 后台执行耗时的任务; 
   * 
   * 方法中的 String 参数对应 AsyncTask的第一个参数; 
   * 返回的 Bitmap 对应的是AsyncTask 的第三个参数; 
   * 
   * 该方法并不运行在UI线程中,主要用于异步操作,可以调用publishProgress()方法触发 
   * onProgressUpdate对UI进行操作; 
   * 
   */ 
  protected Bitmap doInBackground(String... params) { 
    Log.i("msg","doInBackground(String... params)..."); 
 
    try { 
 
      /* 网络访问方式 二 */ 
      /* 
      URL url = new URL(params[0]); 
      HttpsURLConnection cOnnection= (HttpsURLConnection) url.openConnection(); 
      connection.connect(); // 开始连接 
      int zOng= connection.getContentLength(); 
      InputStream is2 = connection.getInputStream(); 
      */ 
 
      /* 开始网络访问数据 */ 
      HttpGet hg = new HttpGet(params[0]); // 此处注意参数的用法 
      HttpClient hc = new DefaultHttpClient(); 
      HttpResponse hr = hc.execute(hg); // 发送请求,得到响应 
 
      // 判断请求是否成功 
      if(hr.getStatusLine().getStatusCode() == HttpStatus.SC_OK){ 
        Log.i("msg", "access success..."); 
        HttpEntity he = hr.getEntity(); 
        InputStream is = he.getContent(); // 获取输入流对象,好比搭桥 
        long total = he.getContentLength(); // 文件的总字节数 
 
        ByteArrayOutputStream baos = new ByteArrayOutputStream(); // 输出流,临时容器,用于装从is中流出的数据 
 
        byte[] buffer = new byte[1024]; // 缓存容器,每次装载1024 个字节数据 
        int len = 0; // 每次读的字节数 
        int curLen = 0 ; // 已读多少数据 
 
        while((len=is.read(buffer))!=-1){ // 当len !=-1 时,也就是还有数据可读 
          Log.i("msg","begin read data..."+len+",total:"+total); 
          baos.write(buffer, 0, len); // 向临时容器中装数据 
          curLen=curLen+len; // 更新已读的数据 
 
          /* 在UI显示当前读取的进度 , 调用次方法触发onProgressUpdate() 方法执行 */ 
          publishProgress((int)(((float)curLen/total)*100)); 
        } 
 
        Bitmap bitmap = BitmapFactory.decodeByteArray(baos.toByteArray(), 0, (int)total); 
        is.close(); 
        return bitmap; 
      } 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
 
    return null; 
  } 
 
  /** 
   * 括号中的参数:String 对应的是AsyncTask 的第三个参数,也就是 
   * 接收了 从doInBackground() 返回的结果; 
   * 此方法在 doInBackground() 方法执行结束后执行,运行在UI线程中, 
   * 可以对UI进行更新 
   */ 
  protected void onPostExecute(Bitmap result) { 
    super.onPostExecute(result); 
    Log.i("msg","onPostExecute(String result)..."+result.getHeight()); 
  } 
 
 
  /** 
   * 方法括号中的Integer 对应AsyncTask 中的第二个参数; 
   * 在doInBackground() 中每次调用publishProgress() 时被执行; 
   * 该方法是在UI线程中的,所以可以用于对UI进行更新 
   */ 
  protected void onProgressUpdate(Integer... values) { 
    super.onProgressUpdate(values); 
 
    Log.i("msg","onProgressUpdate(Integer... values)..."+values[0]); 
  } 
 
 
  /** 
   * 图片的下载 
   */ 
  public HttpURLConnection downPic(String urltemp){ 
 
    try { 
      URL url = new URL(urltemp); // 确定连接地址 
      // 打开一个连接 
      HttpURLConnection cOnnection= (HttpURLConnection) url.openConnection(); 
      connection.connect(); // 开始连接 
      return connection; 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
    return null; 
  } 
 
 
 
} 


以上就是Android AsyncTask的应用实例,如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


推荐阅读
  • 从理想主义者的内心深处萌发的技术信仰,推动了云原生技术在全球范围内的快速发展。本文将带你深入了解阿里巴巴在开源领域的贡献与成就。 ... [详细]
  • 本文由chszs撰写,详细介绍了Apache Mina框架的核心开发流程及自定义协议处理方法。文章涵盖从创建IoService实例到协议编解码的具体步骤,适合希望深入了解Mina框架应用的开发者。 ... [详细]
  • Spring Security基础配置详解
    本文详细介绍了Spring Security的基础配置方法,包括如何搭建Maven多模块工程以及具体的安全配置步骤,帮助开发者更好地理解和应用这一强大的安全框架。 ... [详细]
  • 吴石访谈:腾讯安全科恩实验室如何引领物联网安全研究
    腾讯安全科恩实验室曾两次成功破解特斯拉自动驾驶系统,并远程控制汽车,展示了其在汽车安全领域的强大实力。近日,该实验室负责人吴石接受了InfoQ的专访,详细介绍了团队未来的重点方向——物联网安全。 ... [详细]
  • Fiddler 安装与配置指南
    本文详细介绍了Fiddler的安装步骤及配置方法,旨在帮助用户顺利抓取用户Token。文章还涵盖了一些常见问题的解决方案,以确保安装过程顺利。 ... [详细]
  • 搭建个人博客:WordPress安装详解
    计划建立个人博客来分享生活与工作的见解和经验,选择WordPress是因为它专为博客设计,功能强大且易于使用。 ... [详细]
  • Docker安全策略与管理
    本文探讨了Docker的安全挑战、核心安全特性及其管理策略,旨在帮助读者深入理解Docker安全机制,并提供实用的安全管理建议。 ... [详细]
  • 理解浏览器历史记录(2)hashchange、pushState
    阅读目录1.hashchange2.pushState本文也是一篇基础文章。继上文之后,本打算去研究pushState,偶然在一些信息中发现了锚点变 ... [详细]
  • 精选10款Python框架助力并行与分布式机器学习
    随着神经网络模型的不断深化和复杂化,训练这些模型变得愈发具有挑战性,不仅需要处理大量的权重,还必须克服内存限制等问题。本文将介绍10款优秀的Python框架,帮助开发者高效地实现分布式和并行化的深度学习模型训练。 ... [详细]
  • Android与JUnit集成测试实践
    本文探讨了如何在Android项目中集成JUnit进行单元测试,并详细介绍了修改AndroidManifest.xml文件以支持测试的方法。 ... [详细]
  • 流处理中的计数挑战与解决方案
    本文探讨了在流处理中进行计数的各种技术和挑战,并基于作者在2016年圣何塞举行的Hadoop World大会上的演讲进行了深入分析。文章不仅介绍了传统批处理和Lambda架构的局限性,还详细探讨了流处理架构的优势及其在现代大数据应用中的重要作用。 ... [详细]
  • 本文详细介绍如何在华为鲲鹏平台上构建和使用适配ARM架构的Redis Docker镜像,解决常见错误并提供优化建议。 ... [详细]
  • Flutter 核心技术与混合开发模式深入解析
    本文深入探讨了 Flutter 的核心技术,特别是其混合开发模式,包括统一管理模式和三端分离模式,以及混合栈原理。通过对比不同模式的优缺点,帮助开发者选择最适合项目的混合开发策略。 ... [详细]
  • Tomcat SSL 配置指南
    本文详细介绍了如何在 Tomcat 中配置 SSL,以确保 Web 应用的安全性。通过正确的配置,可以启用 HTTPS 协议并保护数据传输的安全。 ... [详细]
  • 使用Echarts for Weixin 小程序实现中国地图及区域点击事件
    本文介绍了如何使用Echarts for Weixin在微信小程序中构建中国地图,并实现区域点击事件。包括效果展示、条件准备和逻辑实现的具体步骤。 ... [详细]
author-avatar
mobiledu2502871243
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有