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

Android中AsyncTask与handler用法实例分析

这篇文章主要介绍了Android中AsyncTask与handler用法,以实例形式较为详细的分析了Android中AsyncTask与handler的功能、用法与相关注意事项,并附带完整实例源码供读者下载,需要的朋友可以参考下

本文实例讲述了Android中AsyncTask与handler用法。分享给大家供大家参考,具体如下:

首先,我们得明确下一个概念,什么是UI线程。顾名思义,ui线程就是管理着用户界面的那个线程!

android的ui线程操作并不是安全的,并且和用户直接进行界面交互的操作都必须在ui线程中进行才可以。这种模式叫做单线程模式。

我们在单线程模式下编程一定要注意:不要阻塞ui线程、确保只在ui线程中访问ui组件

当我们要执行一个复杂耗时的算法并且最终要将计算结果反映到ui上时,我们会发现,我们根本没办法同时保证上面的两点要求;我们肯定会想到开启一个新的线程,让这个复杂耗时的任务到后台去执行,但是执行完毕了呢?我们发现,我们无法再与ui进行交互了。
为了解决这种情况,android为我们提供了很多办法。

1)、handler和message机制:通过显示的抛出、捕获消息与ui进行交互;

2)、Activity.runOnUiThread(Runnable):如果当前线程为ui线程,则立即执行;否则,将参数中的线程操作放入到ui线程的事件队列中,等待执行。

3)、View.post(Runnable):将操作放入到message队列中,如果放入成功,该操作将会在ui线程中执行,并返回true,否则返回false

4)、View.postDelayed(Runnable, long)跟第三条基本一样,只不过添加了一个延迟时间。

5)、android1.5以后为我们提供了一个工具类来搞定这个问题AsyncTask.

AsyncTask是抽象类,定义了三种泛型类型 Params,Progress,Result。

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

用程序调用,开发者需要做的就是实现这些方法。

1) 子类化AsyncTask
2) 实现AsyncTask中定义的下面一个或几个方法

onPreExecute(),该方法将在执行实际的后台操作前被UI thread调用。可以在该方法中做一些准备工作,如在界面上显示一个进度条。
doInBackground(Params…),将在onPreExecute 方法执行后马上执行,该方法运行在后台线程中。这里将主要负责执行那些很耗时的后台计算工作。可以调用 publishProgress方法来更新实时的任务进度。该方法是抽象方法,子类必须实现。
onProgressUpdate(Progress…),在publishProgress方法被调用后,UI thread将调用这个方法从而在界面上展示任务的进展情况,例如通过一个进度条进行展示。
onPostExecute(Result),在doInBackground 执行完成后,onPostExecute 方法将被UI thread调用,后台的计算结果将通过该方法传递到UI thread.

为了正确的使用AsyncTask类,以下是几条必须遵守的准则:

1) Task的实例必须在UI thread中创建
2) execute方法必须在UI thread中调用
3) 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params…), onProgressUpdate(Progress…)这几个方法
4) 该task只能被执行一次,否则多次调用时将会出现异常

package cn.com.chenzheng_java; 
import android.os.AsyncTask; 
/** 
 * 
 * @author chenzheng_java 
 * @description 异步任务AcyncTask示例 
 *   
 */ 
public class MyAsyncTask extends AsyncTask { 
 /** 
 * 该方法由ui线程进行调用,用户可以在这里尽情的访问ui组件。 
 * 很多时候,我们会在这里显示一个进度条啥的,以示后台正在 
 * 执行某项功能。 
 */ 
 @Override 
 protected void onPreExecute() { 
 super.onPreExecute(); 
 } 
 /** 
 * 该方法由后台进程进行调用,进行主要的耗时的那些计算。 
 * 该方法在onPreExecute方法之后进行调用。当然在执行过程中 
 * 我们可以每隔多少秒就调用一次publishProgress方法,更新 
 * 进度信息 
 */ 
 @Override 
 protected Object doInBackground(String... params) { 
 return null; 
 } 
 /** 
 * doInBackground中调用了publishProgress之后,ui线程就会 
 * 调用该方法。你可以在这里动态的改变进度条的进度,让用户知道 
 * 当前的进度。 
 */ 
 @Override 
 protected void onProgressUpdate(Integer... values) { 
 super.onProgressUpdate(values); 
 } 
 /** 
 * 当doInBackground执行完毕之后,由ui线程调用。可以在这里 
 * 返回我们计算的最终结果给用户。 
 */ 
 @Override 
 protected void onPostExecute(Object result) { 
 super.onPostExecute(result); 
 } 
}

下面介绍最本质的多线程:hanlder和message机制:

为何需要多线程:

在日常应用中,我们通常需要处理一些“后台,用户不可见”的操作,例如说,我们需要下载一个音乐,要是你的应用必须等用户下载完成之后才可以进行别的操作,那肯定让用户非常的不爽。这时候,我们通常的做法是,让这些操作去后台执行,然后等后台执行完毕之后,再给用户弹出相应的提示信息。这时候,我们就需要使用多线程机制,然后通过创建一个新的线程来执行这些操作。

明白了,实现需求,我们就准备着手实现了。但是,经过进一步的了解,我们悲剧的发现,android中的线程机制是,只能在UI线程中和用户进行交互。当我们创建了一个新线程,执行了一些后台操作,执行完成之后,我们想要给用户弹出对话框以确认,但是却悲剧的发现,我们根本无法返回UI主线程了。(UI线程就是你当前看到的这些交互界面所属的线程)。

这时候,我们如果想要实现这些功能,我们就需要一个android为我们提供的handler和message机制。

先讲解下编程机制:

我们通常在UI线程中创建一个handler,handler相当于一个处理器,它主要负责处理和绑定到该handler的线程中的message。每一个handler都必须关联一个looper,并且两者是一一对应的,注意,这点很重要哦!此外,looper负责从其内部的messageQueue中拿出一个个的message给handler进行处理。因为我们这里handler是在UI线程中实现的,所以经过这么一个handler、message机制,我们就可以回到UI线程中了。

handler:处理后台进程返回数据的工作人员。
message:后台进程返回的数据,里面可以存储bundle等数据格式
messageQueue:是线程对应looper的一部分,负责存储从后台进程中抛回的和当前handler绑定的message,是一个队列。
looper:looper相当于一个messageQueue的管理人员,它会不停的循环的遍历队列,然后将符合条件的message一个个的拿出来交给handler进行处理。

注意,handler是在UI线程中声明的,如果我们直接用类似代码执行一个线程的话,实际上并没有创建一个新的线程,因为handler已经跟默认的UI线程中的looper绑定了。

如果有兴趣的话,可以去看下Handler的默认空构造函数便知道原因了,里面直接绑定了当前UI线程的looper。

下面给出一个比较简单,并且实用的实例。

public class MainActivity extends Activity implements OnClickListener { 
  private Button btnTXT; 
  private TextView tvTXT; 
  private StringBuffer returnMsg; 
  @Override 
  public void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_main); 
    btnTXT = (Button)findViewById(R.id.btnTXT); 
    tvTXT = (TextView)findViewById(R.id.tvTXT); 
    btnTXT.setOnClickListener(this);     
  } 
  @Override 
  public void onClick(View v) { 
    returnMsg = new StringBuffer(); 
    // 创建一个包含Looper的线程,这里如果没有HandlerThread的调用,会直接将后边的MyRunnable放到UI线程队列(myHandler.post(new MyRunnable())) 
    HandlerThread handlerThread = new HandlerThread("handler_thread"); 
    handlerThread.start();   // 启动自定义处理线程 
    myHandler = new MyHandler(handlerThread.getLooper());    // 将handler绑定到新线程  
    myHandler.post(new MyRunnable());    // 在新线程中执行任务  
  } 
  /** 主线程Handler,可以与UI控件交互 */ 
  Handler mainHanlder = new Handler(){ 
    @Override 
    public void handleMessage(Message msg) { 
      if(msg.what == 0) { 
        tvTXT.setText(returnMsg.toString());  // 与主线程控件打交道(直接访问) 
      } 
    } 
  }; 
  /** 构造Hanlder,不可与UI控件直接交互 */ 
  private MyHandler myHandler = null; 
  private class MyHandler extends Handler{ 
    /** 
     * 使用默认的构造函数,会将handler绑定当前UI线程的looper。 
     * 如果想使用多线程这里是不能使用默认的构造方法的。 
     */  
    public MyHandler(){ 
      super(); 
    } 
    /** 构造函数,自定义looper */ 
    public MyHandler(Looper looper) { 
      super(looper); 
    } 
    // 处理具体的message消息,继承自父类的方法 
    @Override 
    public void handleMessage(Message msg) { 
      int what = msg.what;   
      Bundle bundle = (Bundle)msg.obj;      // 提取bundle中的信息 
      String name = bundle.getString("name"); 
      String sex = bundle.getString("sex"); 
      boolean marry = bundle.getBoolean("marray"); 
      int age = bundle.getInt("age"); 
      StringBuffer strBuf = new StringBuffer();    // 拼接bundle信息 
      strBuf.append("what = ").append(what).append("\n\n"); 
      strBuf.append("name = ").append(name).append("\n"); 
      strBuf.append("sex = ").append(sex).append("\n"); 
      strBuf.append("marry = ").append(marry).append("\n"); 
      strBuf.append("age = ").append(age).append("\n\n"); 
      strBuf.append("http://blog.csdn.net/sunboy_2050"); 
      returnMsg = returnMsg.append(strBuf);  // 保存要显示的结果 
      mainHanlder.sendEmptyMessage(0);    // 向主线程mainHanlder发送消息,与UI控件交互显示结果 
      super.handleMessage(msg); 
    } 
  } 
  // 构造Runnable,处理后台业务逻辑,如下载 
  private class MyRunnable implements Runnable{ 
    @Override 
    public void run() { 
      try { 
        Message msg = Message.obtain(myHandler);  // 捕获myHandler消息
        msg.what = 10; 
        Bundle bundle = new Bundle();        // 封装bundle信息 
        bundle.putString("name", "yanggang"); 
        bundle.putString("sex", "pure boy"); 
        bundle.putBoolean("marry", false); 
        bundle.putInt("age", 18); 
        msg.obj = bundle; 
        long thID = Thread.currentThread().getId(); 
        returnMsg.append(thID).append(" : send msg start...").append("\n"); 
        msg.sendToTarget();   // 向myHandler发送消息 
        Thread.sleep(3000); 
      } catch (Exception e) { 
        Log.i("", "Runnable send msg error..."); 
        e.printStackTrace(); 
      } 
    } 
  } 
}

运行结果:

完整实例代码代码点击此处本站下载。

希望本文所述对大家Android程序设计有所帮助。


推荐阅读
  • 初探性能优化:入门指南与实践技巧
    在编程领域,常有“尚未精通编码便急于优化”的声音。为了从性能优化的角度提升代码质量,本文将带领读者初步探索性能优化的基本概念与实践技巧。即使程序看似运行良好,数据处理效率仍有待提高,通过系统学习性能优化,能够帮助开发者编写更加高效、稳定的代码。文章不仅介绍了性能优化的基础知识,还提供了实用的调优方法和工具,帮助读者在实际项目中应用这些技术。 ... [详细]
  • 在探讨Hibernate框架的高级特性时,缓存机制和懒加载策略是提升数据操作效率的关键要素。缓存策略能够显著减少数据库访问次数,从而提高应用性能,特别是在处理频繁访问的数据时。Hibernate提供了多层次的缓存支持,包括一级缓存和二级缓存,以满足不同场景下的需求。懒加载策略则通过按需加载关联对象,进一步优化了资源利用和响应时间。本文将深入分析这些机制的实现原理及其最佳实践。 ... [详细]
  • 在Java中,抽象类无法直接实例化的原因在于其设计初衷是为了提供一种模板方法模式,其中包含未实现的方法。这些方法需要由子类来具体实现。因此,直接实例化抽象类没有实际意义,也无法满足编译器的要求。理解这一点有助于更好地利用抽象类进行面向对象编程。 ... [详细]
  • 小王详解:内部网络中最易理解的NAT原理剖析,挑战你的认知极限
    小王详解:内部网络中最易理解的NAT原理剖析,挑战你的认知极限 ... [详细]
  • 第六章:枚举类型与switch结构的应用分析
    第六章深入探讨了枚举类型与 `switch` 结构在编程中的应用。枚举类型(`enum`)是一种将一组相关常量组织在一起的数据类型,广泛存在于多种编程语言中。例如,在 Cocoa 框架中,处理文本对齐时常用 `NSTextAlignment` 枚举来表示不同的对齐方式。通过结合 `switch` 结构,可以更清晰、高效地实现基于枚举值的逻辑分支,提高代码的可读性和维护性。 ... [详细]
  • 作为软件工程专业的学生,我深知课堂上教师讲解速度之快,很多时候需要课后自行消化和巩固。因此,撰写这篇Java Web开发入门教程,旨在帮助初学者更好地理解和掌握基础知识。通过详细记录学习过程,希望能为更多像我一样在基础方面还有待提升的学员提供有益的参考。 ... [详细]
  • 深入解析HTTPS:保障Web安全的加密协议
    本文详细探讨了HTTPS协议在保障Web安全中的重要作用。首先分析了HTTP协议的不足之处,包括数据传输过程中的安全性问题和内容加密的缺失。接着介绍了HTTPS如何通过使用公钥和私钥的非对称加密技术以及混合加密机制,确保数据的完整性和机密性。最后强调了HTTPS的安全性和可靠性,为现代网络通信提供了坚实的基础。 ... [详细]
  • POJ3669题目解析:基于广度优先搜索的详细解答
    POJ3669(http://poj.org/problem?id=3669)是一道典型的广度优先搜索(BFS)问题。由于陨石的降落具有时间属性,导致地图状态会随时间动态变化。因此,可以利用结构体来记录每个陨石的降落时间和位置,从而有效地进行状态更新和路径搜索。 ... [详细]
  • C# .NET 4.1 版本大型信息化系统集成平台中的主从表事务处理标准示例
    在C# .NET 4.1版本的大型信息化系统集成平台中,本文详细介绍了主从表事务处理的标准示例。通过确保所有操作要么全部成功,要么全部失败,实现主表和关联子表的同步插入。主表插入时会返回当前生成的主键,该主键随后用于子表插入时的关联。以下是一个示例代码片段,展示了如何在一个数据库事务中同时添加角色和相关用户。 ... [详细]
  • 本文详细介绍了267 Collections的特性和应用场景。作为Java集合框架中的核心接口,Collection接口是所有单列集合类的顶级接口,涵盖了列表、集合和队列等数据结构。通过具体的应用实例,本文深入解析了Collection接口的各种方法和功能,帮助开发者更好地理解和使用这一重要工具。 ... [详细]
  • 在探讨P1923问题时,我们发现手写的快速排序在最后两个测试用例中出现了超时现象,这在意料之中,因为该题目实际上要求的是时间复杂度为O(n)的算法。进一步研究题解后,发现有选手使用STL中的`nth_element`函数成功通过了所有测试点。本文将详细分析这一现象,并提出相应的优化策略。 ... [详细]
  • 深入浅析JVM垃圾回收机制与收集器概述
    本文基于《深入理解Java虚拟机:JVM高级特性与最佳实践(第3版)》的阅读心得进行整理,详细探讨了JVM的垃圾回收机制及其各类收集器的特点与应用场景。通过分析不同垃圾收集器的工作原理和性能表现,帮助读者深入了解JVM内存管理的核心技术,为优化Java应用程序提供实用指导。 ... [详细]
  • 本文深入探讨了JavaScript中`this`关键字的多种使用方法和技巧。首先,分析了`this`作为全局变量时的行为;接着,讨论了其在对象方法调用中的表现;然后,介绍了`this`在构造函数中的作用;最后,详细解释了通过`apply`等方法改变`this`指向的机制。文章旨在帮助开发者更好地理解和应用`this`关键字,提高代码的灵活性和可维护性。 ... [详细]
  • 探讨LaTeX中四级标题的使用与常见问题解决方案
    在LaTeX文档排版中,四级标题的使用方法及其常见问题的解决策略是本文的重点。通常情况下,LaTeX支持一级、二级和三级标题,分别通过`\section{}`、`\subsection{}`和`\subsubsection{}`命令实现。然而,对于需要四级标题的情况,用户往往面临格式不一致或编译错误等问题。本文将详细介绍如何通过自定义命令或其他扩展包来实现四级标题,并提供具体的示例和解决方案,以帮助用户更好地管理和排版复杂的文档结构。 ... [详细]
  • 链栈虽然通常以数组作为底层实现,但也可以采用链表来构建Stack类。在这种情况下,空堆栈通过NULL指针表示。当新元素被压入堆栈时,它会被添加到链表的头部,从而实现高效的入栈操作。此外,出栈操作则通过移除链表头部的节点来完成,确保了操作的时间复杂度为O(1)。这种设计不仅简化了内存管理,还提高了动态数据处理的灵活性。 ... [详细]
author-avatar
_______王晓
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有