android之后台线程(UI与线程交互)

简介:

本文将讨论android应用程序的线程模型以及如何使用线程来处理耗时较长的操作,而不是在主线程中执行,保证用户界面(UI)的流畅运行。本文还将阐述一些用户界面(UI)中与线程交互的API。

UI用户界面线程
当应用程序启动时,系统会为应用程序创建一个主线程(main)或者叫UI线程,它负责分发事件到不同的组件,包括绘画事件。完成你的应用程序与android UI组件交互。
例如,当您触摸屏幕上的一个按钮时,UI线程会把触摸事件分发到组件上,更改状态并加入事件队列,UI线程会分发请求和通知到各个组件,完成相应的动作。

单线程模型的性能是非常差的,除非你的应用程序相当的简单,特别是当所有的操作都在主线程中执行,比如访问网络或数据库之类的耗时操作将会导致用户界面锁定,所有的事件将不能分发,应用程序就像死了一样,更严重的是当超过5秒时,系统就会弹出“应用程序无响应”的对话框。
如果你想看看什么效果,可以写一个简单的应用程序,在一个Button的OnClickListener中写上Thread.sleep(2000),运行程序你就会看到在应用程序回到正常状态前按钮会保持按下状态2秒,当这种情况发生时,您就会感觉到应用程序反映相当的慢。

总之,我们需要保证主线程(UI线程)不被锁住,如果有耗时的操作,我们需要把它放到一个单独的后台线程中执行。

下面是一个点击按钮后下载一个图片,同时显示到界面的ImageView上的例子:

 
  1. <SPAN>public</SPAN> <SPAN>void</SPAN> onClick<SPAN> 
  2. (</SPAN><SPAN>View</SPAN> v<SPAN>)</SPAN> <SPAN>{</SPAN> 
  3.  <SPAN>new</SPAN> <SPAN>Thread</SPAN><SPAN> 
  4. (</SPAN><SPAN>new</SPAN> <SPAN>Runnable</SPAN><SPAN> 
  5. (</SPAN><SPAN>)</SPAN> <SPAN>{</SPAN>   
  6. <SPAN>public</SPAN> <SPAN>void</SPAN> run<SPAN> 
  7. (</SPAN><SPAN>)</SPAN> <SPAN>{</SPAN> Bitmap b   
  8. <SPAN>=</SPAN> loadImageFromNetwork<SPAN>(</SPAN><SPAN>)  
  9. </SPAN><SPAN>;</SPAN>   
  10. mImageView.<SPAN>setImageBitmap</SPAN><SPAN> 
  11. (</SPAN>b<SPAN>)</SPAN><SPAN>;</SPAN> <SPAN>}</SPAN>   
  12. <SPAN>}</SPAN><SPAN>)</SPAN>.<SPAN>start</SPAN><SPAN> 
  13. (</SPAN><SPAN>)</SPAN><SPAN>;</SPAN> <SPAN>}</SPAN>  

起初,上面的代码似乎是一个很好的解决方案,因为它不会锁住用户界面线程。然面不幸的是,它违反了用户界面单线程模型:android的用户界面工具包不是线程安全的,只能在UI线程中操作它,在上面的代码中,你在一个工作线程中调用mImageView.setImageBitmap(b)时,将会发生意想不到的错误,这种错误是非常难跟踪和调试的。

android提供了几种方法来从其他线程访问UI线程。您可能已经熟悉他们了,下面是一个较全面的列表:

 
  1. Activity.<SPAN>runOnUiThread</SPAN><SPAN>  
  2. (</SPAN><SPAN>Runnable</SPAN><SPAN>)</SPAN>   
  3. <SPAN>View</SPAN>.<SPAN>post</SPAN><SPAN>  
  4. (</SPAN><SPAN>Runnable</SPAN><SPAN>)</SPAN>   
  5. <SPAN>View</SPAN>.<SPAN>postDelayed</SPAN><SPAN>  
  6. (</SPAN><SPAN>Runnable</SPAN>, <SPAN>long</SPAN><SPAN>)  
  7. </SPAN> Handler    

您可以使用这些类和方法中的任何一种纠正前面的代码示例:

 
  1. <SPAN>public</SPAN> <SPAN>void</SPAN> onClick<SPAN>(</SPAN><SPAN>View</SPAN> v<SPAN>)</SPAN> <SPAN>{</SPAN> <SPAN>new</SPAN> <SPAN>Thread</SPAN><SPAN>(</SPAN><SPAN>new</SPAN> <SPAN>Runnable</SPAN><SPAN>(</SPAN><SPAN>)</SPAN> <SPAN>{</SPAN> <SPAN>public</SPAN> <SPAN>void</SPAN> run<SPAN>  
  2. (</SPAN><SPAN>)</SPAN> <SPAN>{</SPAN> <SPAN>final</SPAN>   
  3. Bitmap b <SPAN>=</SPAN> loadImageFromNetwork<SPAN>  
  4. (</SPAN><SPAN>)</SPAN><SPAN>;</SPAN>   
  5. mImageView.<SPAN>post</SPAN><SPAN>  
  6. (</SPAN><SPAN>new</SPAN> <SPAN>Runnable</SPAN><SPAN>  
  7. (</SPAN><SPAN>)</SPAN> <SPAN>{</SPAN>   
  8. <SPAN>public</SPAN> <SPAN>void</SPAN> run<SPAN>  
  9. (</SPAN><SPAN>)</SPAN> <SPAN>{</SPAN>   
  10. mImageView.<SPAN>setImageBitmap</SPAN><SPAN>  
  11. (</SPAN>b<SPAN>)</SPAN><SPAN>;</SPAN> <SPAN>}</SPAN>   
  12. <SPAN>}</SPAN><SPAN>)</SPAN><SPAN>;</SPAN> <SPAN>}  
  13. </SPAN> <SPAN>}</SPAN><SPAN>)  
  14. </SPAN>.<SPAN>start</SPAN><SPAN>(</SPAN><SPAN>)  
  15. </SPAN><SPAN>;</SPAN> <SPAN>}</SPAN>    

不幸的是,这些类和方法也往往使你的代码更复杂,更难以阅读。更糟糕的是,它需要频繁执行复杂的操作界面更新。

为了解决这个问题,1.5和更高版本的Android平台提供了一个实用类称为AsyncTask,简化了长时间运行的任务,需要与用户界面的交互。
类似AsyncTask的一个类UserTask也可用于Android 1.0和1.1版本,它提供了完全相同的API,所有您需要做的是把它的源代码复制到你的应用程序中。

AsyncTask的目标是要为你的线程提供管理服务,我们前面的例子可以很容易的用AsyncTask来改写:

 
  1. <SPAN>public</SPAN> <SPAN>void</SPAN> onClick<SPAN>  
  2. (</SPAN><SPAN>View</SPAN> v<SPAN>)</SPAN> <SPAN>{</SPAN>  
  3.  <SPAN>new</SPAN> DownloadImageTask<SPAN>(</SPAN><SPAN>)  
  4. </SPAN>.<SPAN>execute</SPAN><SPAN>  
  5. (</SPAN><SPAN>"http://www.ideasandroid.com/image.png"</SP  
  6. AN><SPAN>)</SPAN><SPAN>;</SPAN> <SPAN>}</SPAN>   
  7.   <SPAN>private</SPAN> <SPAN>class</SPAN>   
  8. DownloadImageTask <SPAN>extends</SPAN>   
  9. AsyncTask<SPAN><</SPAN>String,   
  10. <SPAN>Void</SPAN>,Bitmap<SPAN>></SPAN> <SPAN>{</SPAN>  
  11.  <SPAN>protected</SPAN> Bitmap doInBackground<SPAN>  
  12. (</SPAN><SPAN>String</SPAN>... <SPAN>urls</SPAN><SPAN>)  
  13. </SPAN> <SPAN>{</SPAN> <SPAN>return</SPAN>   
  14. loadImageFromNetwork<SPAN>(</SPAN>urls<SPAN>  
  15. [</SPAN><SPAN>0</SPAN><SPAN>]</SPAN><SPAN>)  
  16. </SPAN><SPAN>;</SPAN> <SPAN>}</SPAN>     
  17. <SPAN>protected</SPAN> <SPAN>void</SPAN>   
  18. onPostExecute<SPAN>(</SPAN>Bitmap result<SPAN>)</SPAN>  
  19.  <SPAN>{</SPAN>   
  20. mImageView.<SPAN>setImageBitmap</SPAN><SPAN>  
  21. (</SPAN>result<SPAN>)</SPAN><SPAN>;</SPAN> <SPAN>}  
  22. </SPAN> <SPAN>}</SPAN>   

正如你所看到的,我们必须通过继承AsyncTask类来使用它,非常重要的一点是:AsyncTask必须在UI线程中实例化它,并且只能执行一次。

以下是AsyncTask的简要使用方法:
•您可以指定三个参数类型,泛型参数,进度值(执行过程中返回的值)和最终值(执行完返回的值)。
•该方法doInBackground()自动执行工作线程(后台线程)
•onPreExecute(),onPostExecute()和onProgressUpdate()都是在UI线程调用
•由doInBackground返回的值()发送到onPostExecute()
•您可以在执行doInBackground()时调用publishProgress()然后在UI组程中执行onProgressUpdate()。
•您可以从任何线程随时取消任务

不管你是否使用AsyncTask,时刻牢记单一线程模型的两条规则:
1、不要锁住用户界面。
2、确保只在UI线程中访问android用户界面工具包中的组件。

AsyncTask只是可以让你更容易地做这些事情。

本文非原创作品,全是从洋文翻译过来的,如有不对的地方请拍砖!!




     本文转自xyz_lmn51CTO博客,原文链接:,http://blog.51cto.com/xyzlmn/818235如需转载请自行联系原作者




相关文章
|
7月前
|
存储 Android开发 数据安全/隐私保护
Thanox安卓系统增加工具下载,管理、阻止、限制后台每个APP运行情况
Thanox是一款Android系统管理工具,专注于权限、后台启动及运行管理。支持应用冻结、系统优化、UI自定义和模块管理,基于Xposed框架开发,安全可靠且开源免费,兼容Android 6.0及以上版本。
898 4
|
Java Linux Android开发
移动应用开发与操作系统的交互:深入理解Android和iOS
在数字时代,移动应用成为我们日常生活的一部分。本文将深入探讨移动应用开发的核心概念、移动操作系统的工作原理以及它们如何相互作用。我们将通过实际代码示例,展示如何在Android和iOS平台上创建一个简单的“Hello World”应用,并解释其背后的技术原理。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和知识。
|
XML Android开发 UED
"掌握安卓开发新境界:深度解析AndroidManifest.xml中的Intent-filter配置,让你的App轻松响应scheme_url,开启无限交互可能!"
【8月更文挑战第2天】在安卓开发中,scheme_url 通过在`AndroidManifest.xml`中配置`Intent-filter`,使应用能响应特定URL启动或执行操作。基本配置下,应用可通过定义特定URL模式的`Intent-filter`响应相应链接。
442 12
|
Android开发 开发者 Kotlin
Android 多进程情况下判断应用是否处于前台或者后台
本文介绍在多进程环境下判断Android应用前后台状态的方法。通过`ActivityManager`和服务信息`RunningAppProcessInfo`可有效检测应用状态,优化资源使用。提供Kotlin代码示例,帮助开发者轻松集成。
1309 8
|
编解码 网络协议 Android开发
Android平台GB28181设备接入模块实现后台service按需回传摄像头数据到国标平台侧
我们在做Android平台GB28181设备对接模块的时候,遇到这样的技术需求,开发者希望能以后台服务的形式运行程序,国标平台侧没有视频回传请求的时候,仅保持信令链接,有发起视频回传请求或语音广播时,打开摄像头,并实时回传音视频数据或接收处理国标平台侧发过来的语音广播数据。
211 3
|
监控 Android开发 开发者
Android经典面试题之实战经验分享:如何简单实现App的前后台监听判断
本文介绍在Android中判断应用前后台状态的两种方法:`ActivityLifecycleCallbacks`和`ProcessLifecycleOwner`。前者提供精细控制,适用于需针对每个Activity处理的场景;后者简化前后台检测,适用于多数应用。两者各有优劣:`ActivityLifecycleCallbacks`更精确但复杂度高;`ProcessLifecycleOwner`更简便但可能在极端场景下略有差异。根据应用需求选择合适方法。
552 2
|
XML Android开发 UED
💥Android UI设计新风尚!掌握Material Design精髓,让你的界面颜值爆表!🎨
【7月更文挑战第28天】随着移动应用市场的发展,用户对界面设计的要求不断提高。Material Design是由Google推出的设计语言,强调真实感、统一性和创新性,通过模拟纸张和墨水的物理属性创造沉浸式体验。它注重色彩、排版、图标和布局的一致性,确保跨设备的统一视觉风格。Android Studio提供了丰富的Material Design组件库,如按钮、卡片等,易于使用且美观。
672 1
|
设计模式 安全 Java
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
235 1
|
Android开发 开发者
Android UI设计中,Theme定义了Activity的视觉风格,包括颜色、字体、窗口样式等,定义在`styles.xml`。
【6月更文挑战第26天】Android UI设计中,Theme定义了Activity的视觉风格,包括颜色、字体、窗口样式等,定义在`styles.xml`。要更改主题,首先在该文件中创建新主题,如`MyAppTheme`,覆盖所需属性。然后,在`AndroidManifest.xml`中应用主题至应用或特定Activity。运行时切换主题可通过重新设置并重启Activity实现,或使用`setTheme`和`recreate()`方法。这允许开发者定制界面并与品牌指南匹配,或提供多主题选项。
366 6
|
开发工具 Android开发 开发者
Android `.9.png` 图像是用于UI的可拉伸格式,保持元素清晰度和比例
【6月更文挑战第26天】Android `.9.png` 图像是用于UI的可拉伸格式,保持元素清晰度和比例。通过边上的黑线定义拉伸区域,右下角黑点标识内容区域,适应文本或组件大小变化。常用于按钮、背景等,确保跨屏幕尺寸显示质量。Android SDK 提供`draw9patch.bat`工具来创建和编辑。**
438 6