Android应用程序线程消息循环模型分析(3)

简介:
     3. 需要与UI交互的应用程序子线程消息模型
        前面说过,我们开发应用程序的时候,经常中需要创建一个子线程来在后台执行一个特定的计算任务,而在这个任务计算的过程中,需要不断地将计算进度或者计算结果展现在应用程序的界面中。典型的例子是从网上下载文件,为了不阻塞应用程序的主线程,我们开辟一个子线程来执行下载任务,子线程在下载的同时不断地将下载进度在应用程序界面上显示出来,这样做出来程序就非常友好。由于子线程不能直接操作应用程序的UI,因此,这时候,我们就可以通过往应用程序的主线程中发送消息来通知应用程序主线程更新界面上的下载进度。因为类似的这种情景在实际开发中经常碰到,Android系统为开发人员提供了一个异步任务类(AsyncTask)来实现上面所说的功能,即它会在一个子线程中执行计算任务,同时通过主线程的消息循环来获得更新应用程序界面的机会。
        为了更好地分析AsyncTask的实现,我们先举一个例子来说明它的用法。在前面一篇文章 Android系统中的广播(Broadcast)机制简要介绍和学习计划 中,我们开发了一个应用程序Broadcast,其中使用了AsyncTask来在一个线程在后台在执行计数任务,计数过程通过广播(Broadcast)来将中间结果在应用程序界面上显示出来。在这个例子中,使用广播来在应用程序主线程和子线程中传递数据不是最优的方法,当时只是为了分析Android系统的广播机制而有意为之的。在本节内容中,我们稍微这个例子作一个简单的修改,就可以通过消息的方式来将计数过程的中间结果在应用程序界面上显示出来。
        为了区别 Android系统中的广播(Broadcast)机制简要介绍和学习计划 一文中使用的应用程序Broadcast,我们将本节中使用的应用程序命名为Counter。首先在Android源代码工程中创建一个Android应用程序工程,名字就为Counter,放在packages/experimental目录下。关于如何获得Android源代码工程,请参考 在Ubuntu上下载、编译和安装Android最新源代码 一文;关于如何在Android源代码工程中创建应用程序工程,请参考 在Ubuntu上为Android系统内置Java应用程序测试Application Frameworks层的硬件服务 一文。这个应用程序工程定义了一个名为shy.luo.counter的package,这个例子的源代码主要就是实现在这个目录下的Counter.java文件中:
 
 
  1. package shy.luo.counter;   
  2.    
  3. import android.app.Activity;   
  4. import android.content.ComponentName;   
  5. import android.content.Context;   
  6. import android.content.Intent;   
  7. import android.content.IntentFilter;   
  8. import android.os.Bundle;   
  9. import android.os.AsyncTask;   
  10. import android.util.Log;   
  11. import android.view.View;   
  12. import android.view.View.OnClickListener;   
  13. import android.widget.Button;   
  14. import android.widget.TextView;   
  15.    
  16. public class Counter extends Activity implements OnClickListener {   
  17.     private final static String LOG_TAG = "shy.luo.counter.Counter";   
  18.    
  19.     private Button startButton = null;   
  20.     private Button stopButton = null;   
  21.     private TextView counterText = null;   
  22.    
  23.     private AsyncTask<Integer, Integer, Integer> task = null;   
  24.     private boolean stop = false;   
  25.    
  26.     @Override   
  27.     public void onCreate(Bundle savedInstanceState) {   
  28.         super.onCreate(savedInstanceState);   
  29.         setContentView(R.layout.main);   
  30.    
  31.         startButton = (Button)findViewById(R.id.button_start);   
  32.         stopButton = (Button)findViewById(R.id.button_stop);   
  33.         counterText = (TextView)findViewById(R.id.textview_counter);   
  34.    
  35.         startButton.setOnClickListener(this);   
  36.         stopButton.setOnClickListener(this);   
  37.    
  38.         startButton.setEnabled(true);   
  39.         stopButton.setEnabled(false);   
  40.    
  41.    
  42.         Log.i(LOG_TAG, "Main Activity Created.");   
  43.     }   
  44.    
  45.    
  46.     @Override   
  47.     public void onClick(View v) {   
  48.         if(v.equals(startButton)) {   
  49.             if(task == null) {   
  50.                 task = new CounterTask();   
  51.                 task.execute(0);   
  52.    
  53.                 startButton.setEnabled(false);   
  54.                 stopButton.setEnabled(true);   
  55.             }   
  56.         } else if(v.equals(stopButton)) {   
  57.             if(task != null) {   
  58.                 stop = true;   
  59.                 task = null;   
  60.    
  61.                 startButton.setEnabled(true);   
  62.                 stopButton.setEnabled(false);   
  63.             }   
  64.         }   
  65.     }   
  66.    
  67.     class CounterTask extends AsyncTask<Integer, Integer, Integer> {   
  68.         @Override   
  69.         protected Integer doInBackground(Integer... vals) {   
  70.             Integer initCounter = vals[0];   
  71.    
  72.             stop = false;   
  73.             while(!stop) {   
  74.                 publishProgress(initCounter);   
  75.    
  76.                 try {   
  77.                     Thread.sleep(1000);   
  78.                 } catch (InterruptedException e) {   
  79.                     e.printStackTrace();   
  80.                 }   
  81.    
  82.                 initCounter++;   
  83.             }   
  84.    
  85.             return initCounter;   
  86.         }   
  87.    
  88.         @Override   
  89.         protected void onProgressUpdate(Integer... values) {   
  90.             super.onProgressUpdate(values);   
  91.    
  92.             String text = values[0].toString();   
  93.             counterText.setText(text);   
  94.         }   
  95.    
  96.         @Override   
  97.         protected void onPostExecute(Integer val) {   
  98.             String text = val.toString();   
  99.             counterText.setText(text);   
  100.         }   
  101.     };   
  102. }   
        这个计数器程序很简单,它在界面上有两个按钮Start和Stop。点击Start按钮时,便会创建一个CounterTask实例task,然后调用它的execute函数就可以在应用程序中启动一个子线程,并且通过调用这个CounterTask类的doInBackground函数来执行计数任务。在计数的过程中,会通过调用publishProgress函数来将中间结果传递到onProgressUpdate函数中去,在onProgressUpdate函数中,就可以把中间结果显示在应用程序界面了。点击Stop按钮时,便会通过设置变量stop为true,这样,CounterTask类的doInBackground函数便会退出循环,然后将结果返回到onPostExecute函数中去,在onPostExecute函数,会把最终计数结果显示在用程序界面中。
 
       在这个例子中,我们需要注意的是:
       A. CounterTask类继承于AsyncTask类,因此它也是一个异步任务类;
       B. CounterTask类的doInBackground函数是在后台的子线程中运行的,这时候它不可以操作应用程序的界面;
       C. CounterTask类的onProgressUpdate和onPostExecute两个函数是应用程序的主线程中执行,它们可以操作应用程序的界面。
       关于C这一点的实现原理,我们在后面会分析到,这里我们先完整地介绍这个例子,以便读者可以参考做一下实验。




本文转自 Luoshengyang 51CTO博客,原文链接:http://blog.51cto.com/shyluo/966881,如需转载请自行联系原作者
目录
相关文章
|
2月前
|
开发框架 前端开发 Android开发
Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势
本文深入探讨了 Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势。这对于实现高效的跨平台移动应用开发具有重要指导意义。
190 4
|
2月前
|
安全 Android开发 数据安全/隐私保护
深入探讨iOS与Android系统安全性对比分析
在移动操作系统领域,iOS和Android无疑是两大巨头。本文从技术角度出发,对这两个系统的架构、安全机制以及用户隐私保护等方面进行了详细的比较分析。通过深入探讨,我们旨在揭示两个系统在安全性方面的差异,并为用户提供一些实用的安全建议。
|
27天前
|
JSON Java API
探索安卓开发:打造你的首个天气应用
在这篇技术指南中,我们将一起潜入安卓开发的海洋,学习如何从零开始构建一个简单的天气应用。通过这个实践项目,你将掌握安卓开发的核心概念、界面设计、网络编程以及数据解析等技能。无论你是初学者还是有一定基础的开发者,这篇文章都将为你提供一个清晰的路线图和实用的代码示例,帮助你在安卓开发的道路上迈出坚实的一步。让我们一起开始这段旅程,打造属于你自己的第一个安卓应用吧!
56 14
|
30天前
|
Java Linux 数据库
探索安卓开发:打造你的第一款应用
在数字时代的浪潮中,每个人都有机会成为创意的实现者。本文将带你走进安卓开发的奇妙世界,通过浅显易懂的语言和实际代码示例,引导你从零开始构建自己的第一款安卓应用。无论你是编程新手还是希望拓展技术的开发者,这篇文章都将为你打开一扇门,让你的创意和技术一起飞扬。
|
1月前
|
Java 开发工具 Android开发
安卓与iOS开发环境对比分析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自占据半壁江山。本文深入探讨了这两个平台的开发环境,从编程语言、开发工具到用户界面设计等多个角度进行比较。通过实际案例分析和代码示例,我们旨在为开发者提供一个清晰的指南,帮助他们根据项目需求和个人偏好做出明智的选择。无论你是初涉移动开发领域的新手,还是寻求跨平台解决方案的资深开发者,这篇文章都将为你提供宝贵的信息和启示。
30 8
|
28天前
|
搜索推荐 前端开发 测试技术
打造个性化安卓应用:从设计到开发的全面指南
在这个数字时代,拥有一个定制的移动应用不仅是一种趋势,更是个人或企业品牌的重要延伸。本文将引导你通过一系列简单易懂的步骤,从构思你的应用理念开始,直至实现一个功能齐全的安卓应用。无论你是编程新手还是希望拓展技能的开发者,这篇文章都将为你提供必要的工具和知识,帮助你将创意转化为现实。
|
1月前
|
调度 开发者
核心概念解析:进程与线程的对比分析
在操作系统和计算机编程领域,进程和线程是两个基本而核心的概念。它们是程序执行和资源管理的基础,但它们之间存在显著的差异。本文将深入探讨进程与线程的区别,并分析它们在现代软件开发中的应用和重要性。
56 4
|
1月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
2月前
|
API Android开发 iOS开发
深入探索Android与iOS的多线程编程差异
在移动应用开发领域,多线程编程是提高应用性能和响应性的关键。本文将对比分析Android和iOS两大平台在多线程处理上的不同实现机制,探讨它们各自的优势与局限性,并通过实例展示如何在这两个平台上进行有效的多线程编程。通过深入了解这些差异,开发者可以更好地选择适合自己项目需求的技术和策略,从而优化应用的性能和用户体验。
|
2月前
|
安全 Android开发 数据安全/隐私保护
深入探索Android与iOS系统安全性的对比分析
在当今数字化时代,移动操作系统的安全已成为用户和开发者共同关注的重点。本文旨在通过比较Android与iOS两大主流操作系统在安全性方面的差异,揭示两者在设计理念、权限管理、应用审核机制等方面的不同之处。我们将探讨这些差异如何影响用户的安全体验以及可能带来的风险。
41 1