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,如需转载请自行联系原作者
目录
相关文章
|
开发框架 前端开发 Android开发
Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势
本文深入探讨了 Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势。这对于实现高效的跨平台移动应用开发具有重要指导意义。
1596 4
|
存储 Android开发
如何查看Flutter应用在Android设备上已被撤销的权限?
如何查看Flutter应用在Android设备上已被撤销的权限?
708 64
|
存储 Android开发 数据安全/隐私保护
如何在Android设备上撤销Flutter应用程序的所有权限?
如何在Android设备上撤销Flutter应用程序的所有权限?
897 64
|
缓存 Android开发 开发者
Flutter环境配置完成后,如何在Android设备上运行Flutter应用程序?
Flutter环境配置完成后,如何在Android设备上运行Flutter应用程序?
2579 62
|
开发工具 Android开发 开发者
在Android设备上运行Flutter应用程序时,如果遇到设备未授权的问题该如何解决?
在Android设备上运行Flutter应用程序时,如果遇到设备未授权的问题该如何解决?
978 61
|
算法 Java 数据库
Android 应用的主线程在什么情况下会被阻塞?
【10月更文挑战第20天】为了避免主线程阻塞,我们需要合理地设计和优化应用的代码。将耗时操作移到后台线程执行,使用异步任务、线程池等技术来提高应用的并发处理能力。同时,要注意避免出现死循环、不合理的锁使用等问题。通过这些措施,可以确保主线程能够高效地运行,提供流畅的用户体验。
1079 156
|
前端开发 Java Shell
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
982 20
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
Dart 前端开发 Android开发
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
510 4
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
JSON Java API
探索安卓开发:打造你的首个天气应用
在这篇技术指南中,我们将一起潜入安卓开发的海洋,学习如何从零开始构建一个简单的天气应用。通过这个实践项目,你将掌握安卓开发的核心概念、界面设计、网络编程以及数据解析等技能。无论你是初学者还是有一定基础的开发者,这篇文章都将为你提供一个清晰的路线图和实用的代码示例,帮助你在安卓开发的道路上迈出坚实的一步。让我们一起开始这段旅程,打造属于你自己的第一个安卓应用吧!
419 14
|
Java Linux 数据库
探索安卓开发:打造你的第一款应用
在数字时代的浪潮中,每个人都有机会成为创意的实现者。本文将带你走进安卓开发的奇妙世界,通过浅显易懂的语言和实际代码示例,引导你从零开始构建自己的第一款安卓应用。无论你是编程新手还是希望拓展技术的开发者,这篇文章都将为你打开一扇门,让你的创意和技术一起飞扬。
308 13

热门文章

最新文章