热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android多线程之进程与线程

前言对于Android程序中,使用多线程的技术是必不可少的,就拿之前最简单的例子来说明,对于Android4.0+的应用而言,访问网络必须另起线程才可以访问。本片博客介绍Andr

前言

  对于Android程序中,使用多线程的技术是必不可少的,就拿之前最简单的例子来说明,对于Android4.0+的应用而言,访问网络必须另起线程才可以访问。本片博客介绍Android下进程和线程,以及它们的特点及应用,并通过几个Demo来展示Android中简单的线程中操作UI线程的组件。

进程与线程

  一般来说,Android中为一个应用程序开启一个进程进行执行,在这个应用程序中的所有组件,通过单独的线程进行执行,而其中所有的线程,共享该应用程序进程的资源。当一个应用程序启动的时候,Android系统启动一个新的Linux应用程序的进程和一个执行线程。默认情况下,一个应用程序运行中的所有组件运行在相同的进程和线程中,这里的线程一般称为主线程。如果一个应用程序的组件开始的时候,已经存在一个进程,那么应用程序会在与它相同的执行线程中开始这个组件。

进程

  默认情况下,同一应用程序下的所有组件运行在同一进程中,大多数应用程序不应该改变这个。然而,如果需要控制那个进程属于那个组件,可以在AndroidManifest.xml文件中进行配置。一般来说,组件元素:均支持一个android:process属性,可以设置这个属性让不同的组件单独运行在自己的进程中,也可以使用这个属性使不同的应用程序组件运行在相同的进程中,并共享相同的Linux用户ID和赋予同样的证书。

  Tips:元素也支持android:process属性,用于设置所有的组件。

  Android在内存较低的情况下,会关闭一些优先级较低的进程以增大内存运行更重要的进程,而在这个进程中的所有线程,也会被同时销毁。在内存足够的情况下,Android系统会视图尽可能保持应用程序进程,以达到下次的运行的快速启动,但最终需要移除旧的进程,回收内存用于新的或更重要的进程。通过进程的优先级来判断是否被回收,一般会回收优先级低的进程,以给优先级高的进程腾出资源。

  下面是五类Android进程,他们的优先级顺序排列:

  1. Foreground process:前台进程。
  2. Visible prcess:可见进程。
  3. Service process:服务进程。
  4. Background process:后台进程。
  5. Empty process:空进程。

  Tips:一个进程的优先级是可以变化的。

 

线程

  当应用程序启动时,系统会创建一个执行线程在这个应用程序的的进程中,一般被称为“主线程”。这个线程是非常重要的,因为它负责把事件分发给响应的用户组件,包括绘制事件等,因此主线程又被称为UI线程。系统并不会为每个组件创建一个单独的线程,而是在UI线程中,完成这些组件的初始化的,因此系统回调方法是运行在UI线程中,如click事件。

  当程序执行比较复杂的工作来应对用户交互的时候,哪怕应用程序被正确的执行了,单线程模式也可能会导致运行性能很低下。举例来说,如果一切的应用功能都发生在UI线程中,当执行耗时操作的时候,如访问网络或查询数据,均会阻塞UI先,将导致其他的事件不被分发到事件队列中,包括屏幕绘制事件。导致从用户的角度来看,应用程序死掉了。而在Android系统中,当UI线程被阻塞超过几秒钟(大约是5秒)的时候,会弹出“应用程序没有响应”的对话框,造成用户体验差,可能会迫使用户决定退出你的应用或者干脆直接卸载它。

  此外,Android的UI ToolKit包下的所有组件都不是线程安全的,所以,不能在一个单独的工作线程中操作这些UI组件,必须在UI线程中操作。因此,对于单线程模型,Android有两个规则:

  1. 不能阻塞UI线程
  2. 不能在工作线程中访问Android UI ToolKit包下的组件。

  对于耗时的操作,应该放在单独的线程中。例如:下面通过一个Demo监听按钮点击事件,下载一个图片,从单独的线程中,并显示在一个ImageView中。

 1         btnError2.setOnClickListener(new View.OnClickListener() {            
 2             @Override
 3             public void onClick(View v) {
 4                 // 增加一个线程访问网络
 5                 new Thread(new Runnable() {
 6                     @Override
 7                     public void run() {
 8                         // 获取地址下的图片
 9                         Bitmap btm=loadImageFromNetwork("http://ww4.sinaimg.cn/bmiddle/786013a5jw1e7akotp4bcj20c80i3aao.jpg");
10                         imageView1.setImageBitmap(btm);                        
11                     }
12                 }).start();
13                 
14             }
15         });

  起初,这似乎是合理的,启动了一个新线程来访问网络,但是它违反了规则二,不能在Android UI主线程之外修改UI组件,而在click中new Thread的是一个工作线程,在工作线程中无法操作UI组件,以上Demo会报错。

  要修正上面的错误,Android提供几种方法可以从其他线程中访问UI线程:

  • Activity.runOnUiThread(Runnable):运行在指定的UI线程上,如果当前线程是UI线程,那么立即执行,如果当前线程不是UI线程,则发布到UI线程的事件队列中。
  • View.post(Runnable):将事件发布到UI线程中,立即被执行。
  • View.postDelayed(Runnanle,long):将事件发布到UI线程中,延迟被执行,延迟数为传递的long参数。

  下面通过两个Dem来通过上面介绍的方法来操作UI组件:

  Activity.runOnUiThread:

 1         btnRunOnUiThread.setOnClickListener(new View.OnClickListener() {
 2             
 3             @Override
 4             public void onClick(View v) {
 5                 new Thread(new Runnable() {
 6                     
 7                     @Override
 8                     public void run() {
 9                         final Bitmap btm=loadImageFromNetwork("http://ww4.sinaimg.cn/bmiddle/786013a5jw1e7akotp4bcj20c80i3aao.jpg");
10                         MainActivity.this.runOnUiThread(new Runnable() {                            
11                             @Override
12                             public void run() {
13                                 imageView1.setImageBitmap(btm);                                
14                             }
15                         });
16                     }
17                 }).start();
18             }
19         });

  效果演示:

   View.post

 1         btnPost.setOnClickListener(new View.OnClickListener() {
 2             
 3             @Override
 4             public void onClick(View v) {
 5                 new Thread(new Runnable() {
 6                     
 7                     @Override
 8                     public void run() {
 9                         final Bitmap btm=loadImageFromNetwork("http://ww1.sinaimg.cn/bmiddle/88ff29e8jw1e7pjnpfxbrj20dp0a90tb.jpg");
10                         imageView1.post(new Runnable() {
11                             
12                             @Override
13                             public void run() {
14                                 // TODO Auto-generated method stub
15                                 imageView1.setImageBitmap(btm);
16                             }
17                         });
18                     }
19                 }).start();                
20             }
21         });

  效果演示:

  以上Demo中,通过访问网络获取图片的方法,通过HttpClient实现,不清楚的朋友可以参见博客:HttpClient。

  源码下载

 

总结

  虽然上面介绍了几种方式在工作线程中把消息发布到UI线程的消息队列的方式来访问UI组件。但是一般实际的开发当中,这种代码会变的复杂且难于维护。处理更复杂的线程间交互,可以考虑使用Handle+Message,在UI线程中处理工作线程发送过来的消息,还可以继承AsyncTask类来简化工作线程发送消息到主线程中交互UI组件。这两种方式会在接下来的博客中介绍到。

 

 


推荐阅读
  • Android工程师面试准备及设计模式使用场景
    本文介绍了Android工程师面试准备的经验,包括面试流程和重点准备内容。同时,还介绍了建造者模式的使用场景,以及在Android开发中的具体应用。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • Android系统源码分析Zygote和SystemServer启动过程详解
    本文详细解析了Android系统源码中Zygote和SystemServer的启动过程。首先介绍了系统framework层启动的内容,帮助理解四大组件的启动和管理过程。接着介绍了AMS、PMS等系统服务的作用和调用方式。然后详细分析了Zygote的启动过程,解释了Zygote在Android启动过程中的决定作用。最后通过时序图展示了整个过程。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • Nginx使用(server参数配置)
    本文介绍了Nginx的使用,重点讲解了server参数配置,包括端口号、主机名、根目录等内容。同时,还介绍了Nginx的反向代理功能。 ... [详细]
  • 基于layUI的图片上传前预览功能的2种实现方式
    本文介绍了基于layUI的图片上传前预览功能的两种实现方式:一种是使用blob+FileReader,另一种是使用layUI自带的参数。通过选择文件后点击文件名,在页面中间弹窗内预览图片。其中,layUI自带的参数实现了图片预览功能。该功能依赖于layUI的上传模块,并使用了blob和FileReader来读取本地文件并获取图像的base64编码。点击文件名时会执行See()函数。摘要长度为169字。 ... [详细]
  • 本文介绍了在Vue项目中如何结合Element UI解决连续上传多张图片及图片编辑的问题。作者强调了在编码前要明确需求和所需要的结果,并详细描述了自己的代码实现过程。 ... [详细]
  • 欢乐的票圈重构之旅——RecyclerView的头尾布局增加
    项目重构的Git地址:https:github.comrazerdpFriendCircletreemain-dev项目同步更新的文集:http:www.jianshu.comno ... [详细]
  • Week04面向对象设计与继承学习总结及作业要求
    本文总结了Week04面向对象设计与继承的重要知识点,包括对象、类、封装性、静态属性、静态方法、重载、继承和多态等。同时,还介绍了私有构造函数在类外部无法被调用、static不能访问非静态属性以及该类实例可以共享类里的static属性等内容。此外,还提到了作业要求,包括讲述一个在网上商城购物或在班级博客进行学习的故事,并使用Markdown的加粗标记和语句块标记标注关键名词和动词。最后,还提到了参考资料中关于UML类图如何绘制的范例。 ... [详细]
  • 基于Socket的多个客户端之间的聊天功能实现方法
    本文介绍了基于Socket的多个客户端之间实现聊天功能的方法,包括服务器端的实现和客户端的实现。服务器端通过每个用户的输出流向特定用户发送消息,而客户端通过输入流接收消息。同时,还介绍了相关的实体类和Socket的基本概念。 ... [详细]
  • 如何自行分析定位SAP BSP错误
    The“BSPtag”Imentionedintheblogtitlemeansforexamplethetagchtmlb:configCelleratorbelowwhichi ... [详细]
  • 本文介绍了在Python3中如何使用选择文件对话框的格式打开和保存图片的方法。通过使用tkinter库中的filedialog模块的asksaveasfilename和askopenfilename函数,可以方便地选择要打开或保存的图片文件,并进行相关操作。具体的代码示例和操作步骤也被提供。 ... [详细]
  • Python瓦片图下载、合并、绘图、标记的代码示例
    本文提供了Python瓦片图下载、合并、绘图、标记的代码示例,包括下载代码、多线程下载、图像处理等功能。通过参考geoserver,使用PIL、cv2、numpy、gdal、osr等库实现了瓦片图的下载、合并、绘图和标记功能。代码示例详细介绍了各个功能的实现方法,供读者参考使用。 ... [详细]
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 本文介绍了操作系统的定义和功能,包括操作系统的本质、用户界面以及系统调用的分类。同时还介绍了进程和线程的区别,包括进程和线程的定义和作用。 ... [详细]
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社区 版权所有