• 前言
  • Android 开发中多线程的必要性
  • 理解 Android 多线程
    • MessageQueue
    • Looper
    • Handler
    • HandlerThread
  • Android 中为什么只允许在主线程更新 UI
    • Android 系统为了避免过度复杂的线程安全问题特地规定只允许在主线程中更新 UI
    • 而开发者为了避免上述问题需要注意的是
  • 总结
  • Thanks

前言

Android Performance Patterns Season 5 主要介绍了 Android 多线程环境下的性能问题。通过介绍 Android 提供的多种多线程工具类 (AsyncTask, HandlerThread, IntentService, ThreadPool),让我们熟悉各个组件的适用场景,从而在特定场景下选择性能最好的一个。

Android 开发中多线程的必要性

Android 开发中,许多操作都需要由 主线程(UI 线程)来执行,比如:

  • 系统事件(例如设备状态变动)
  • 输入事件
  • 服务
  • 闹钟
  • UI 绘制

我们经常需要针对这些情况编写代码。

由于主线程只有一个,所有任务都是串行执行,如果我们在某个操作中包含大量的网络请求、I/O,将会影响后续用户后续操作。

用户感知最明显的就是界面绘制、响应是否及时:

我们知道 Android 系统的屏幕刷新频率为 60 fps, 也就是每隔 16 ms 刷新一次。如果在某次绘制过程中,我们的操作不能在 16 ms 内完成,那它则不能赶上这次的绘制公交车,只能等下一轮,这种现象叫做 “掉帧”,用户看到的就是界面绘制不连续、卡顿。

为了避免耗时较久的操作导致 “掉帧”,我们会把这些操作从主线程执行换到子线程,这样主线程的其他操作不会受到影响,用户体验也会流畅许多。

理解 Android 多线程

一个线程,主要有三个状态:开始、执行任务、结束。

当线程存活期间,我们会让它执行大量的任务,当任务完成或者主动取消时,线程功成身退。

很多情况下,我们会有很多线程同时存活、执行任务,这时需要添加一个 任务队列,让线程不停地从队列中获取任务,同时有其他线程向其中添加任务,典型的 生产者-消费者 模型:

如果我们来实现这个模型,需要写三个角色:生产者线程、消费者线程、任务队列,同时还要保证它们的协作有条不紊,这可能会难倒一大堆人。

为了让开发者更省心,Android 系统替我们实现了上述类,分别是:

  • MessageQueue
  • Looper
  • Handler

MessageQueue

MessageQueue 就是任务队列,保存着不同类型任务的载体 (Intent, Runnable, Message)。

Looper

Looper 就是我们所说的 “消费者”,它不停地从任务队列中获取任务并执行。

Handler

Handler 就是 “生产者”,它把任务从其他线程送到 MessageQueue 中。

Handler 可以指定任务在任务队列中的位置,也可以按照一定的时间延迟送货。

HandlerThread

HandlerThread 就是上述三个组件的组合。

每个应用启动时,系统会创建一个该应用的进程以及主线程,这里的主线程就是一个 HandlerThread。

这个主线程会处理主要事件,具体内容如图所示:

Android 中为什么只允许在主线程更新 UI

Android 系统中,默认只能在 主线程(UI 线程)更新 UI,当你在 子线程进行 UI 修改时,可能不起作用甚至是奔溃:

为什么要这样设计呢?

我们知道,多线程并发访问资源要遵循重要的原则就是 原子性、可见性、有序性。没有同步机制的情况下,多个线程同时读写内存可能会导致意料之外的问题:

多线程同时操作 UI 也一样,如果想要允许多个线程更新 UI,就要设计对应的同步机制,为了避免这种问题,Android 系统直接规定只允许在 UI 线程更新 UI。

除了线程安全外,还有个原因: UI 组件的生命周期并不确定。

可能有这种情况:我们在某个执行网络请求的线程中持有一个 Button 的引用,然而在请求结果返回之前,这个 button 被 View Hierarchy 移除,这时对 button 的任何操作都不可用,并且也没有意义了。

此外还有一点 线程引用导致的内存泄漏问题。

我们知道每个 View 都持有当前 Context, Activity 的引用,如果子线程持有某个 View 的引用,继而持有了对应 Activity 引用,那么在线程返回之前,即使该 Activity 不可用,也无法回收,这就造成了 内存泄漏。

除了持有 View,线程隐式持有 Activity 也可能导致内存泄漏,只要子线程没有结束,引用关系就一直存在。

比如在 Activity 中创建个内部 AsyncTask:

或者是常见的在 Activity 里创建个 Handler:

正如 Android Studio 提示的那样,内部线程工具类持有外部类引用,可能会导致 内存泄漏。

Android 系统为了避免过度复杂的线程安全问题,特地规定只允许在主线程中更新 UI。

而开发者,为了避免上述问题,需要注意的是:

不要再任何子线程持有 UI 组件或者 Activity 的引用。

总结

本文大概介绍了 Android 中多线程的必要性以及一些基础概念。

Android 系统为我们提供了以下几种工具类:

  • AsyncTask

    • 主线程、子线程间任务的切换
  • HandlerThread 
    • 为某个任务/回调单独开一个线程
  • ThreadPool 
    • 管理多个线程,并发执行任务
  • IntentService 
    • 在子线程中获取 Intent,用于执行由 UI 出发的后台 Service

接下来我们将跟随官方视频逐渐了解这几个工具类的特点,从而能够在合适的场景下选择对的人,尽可能地优化应用的性能。

Android 系统性能优化(45)---Android 多线程相关推荐

  1. Android 系统性能优化(15)---Android性能优化典范 - 第3季

    Android性能优化典范的课程最近更新到第三季了,这次一共12个短视频课程,包括的内容大致有:更高效的ArrayMap容器,使用Android系统提供的特殊容器来避免自动装箱,避免使用枚举类型,注意 ...

  2. Android 系统性能优化(14)---Android性能优化典范 - 第2季

    1)Battery Drain and Networking 对于手机程序,网络操作相对来说是比较耗电的行为.优化网络操作能够显著节约电量的消耗.在性能优化第1季里面有提到过,手机硬件的各个模块的耗电 ...

  3. 那些年,我们一起经历过的 Android 系统性能优化

    Android 系统性能优化,最近几年 ,Google 之外,绝对是中国的手机厂商参与得最深最广.不光深挖系统.场景,还要和 Top 应用合作优化,和各种流氓应用勾心斗角.同时 Google 也在不断 ...

  4. 【书评】一本Android系统性能优化的新书

    Android性能优化,是一个合格的Android程序员必备的技能,现如今几乎所有的Android面试内容都会或多或少涉及性能优化方面的话题. 学习Android性能优化可以让我们在简历上展示自己的专 ...

  5. 【Android CPU 优化】Android CPU 调优 ( Trace 文件分析 | Android Profiler 工具 | CPU Profiler 工具 )

    文章目录 一.Android CPU 优化 二.CPU Profiler 工具 三.相关资源 一.Android CPU 优化 在 Android 中 , 出现 动画掉帧 , 页面切换白屏 , 卡顿 ...

  6. 【Android 内存优化】Android 工程中使用 libjpeg-turbo 压缩图片 ( 初始化压缩对象 | 打开文件 | 设置压缩参数 | 写入压缩图像数据 | 完成压缩 | 释放资源 )

    文章目录 一.使用 libjpeg-turbo 压缩图片流程 二.初始化 JPEG 压缩对象 三.打开文件 四.设置压缩参数 五.开始压缩 六.循环写入压缩数据 七.完成图片压缩及收尾 八.libjp ...

  7. 【Android 内存优化】Android 原生 API 图片压缩原理 ( 哈夫曼编码开关 | 哈夫曼编码原理 | libjpeg-turbo 函数库 )

    文章目录 一. 哈夫曼编码开关 二. 哈夫曼编码原理 三. libjpeg-turbo 函数库 四. libjpeg-turbo 函数库下载 [Android 内存优化]图片文件压缩 ( Androi ...

  8. 【Android 内存优化】Android 原生 API 图片压缩原理 ( Bitmap_compress 方法解析 | Skia 二维图形库 | libjpeg 函数库 | libpng 函数库 )

    文章目录 一. 图片质量压缩方法 二. Skia 二维图形库 三. libjpeg.libpng 函数库引入 在博客 [Android 内存优化]图片文件压缩 ( Android 原生 API 提供的 ...

  9. 【Android 内存优化】Android 原生 API 图片压缩原理 ( 图片质量压缩方法 | 查找 Java 源码中的 native 方法对应的 C++ 源码 )

    文章目录 一. 图片质量压缩方法 二. 查找对应的 Native 方法源码 三. 分析 Bitmap.cpp 中动态注册 Native 方法 在博客 [Android 内存优化]图片文件压缩 ( An ...

  10. Android系统性能优化(69)---含内存优化、布局优化

    Android性能优化:含内存优化.布局优化 前言 在 Android开发中,性能优化策略十分重要 因为其决定了应用程序的开发质量:可用性.流畅性.稳定性等,是提高用户留存率的关键 本文全面讲解性能优 ...

最新文章

  1. 实现一个基于 SharePoint 2013 的 Timecard 应用(下)
  2. php入门时间,php入门教程(二十一) php日期与时间函数
  3. javascript中闭包的一些理解
  4. Android 手把手教您自定义ViewGroup
  5. HTML5性能优化需要注意的几个问题
  6. 寒假与春节终归,新学期和新任务又至
  7. 请接受这份货真价“无”的PPT
  8. java ee cdi_Java EE CDI限定词:快速浏览
  9. 服务器ping不通的解决办法之阿里云云服务器VNC报错Failed to execute /sbin/init
  10. 素数的有关性质(二)欧拉函数的一些定理证明与计算
  11. 利用jmeter做一个简单的性能测试并进行参数化设置
  12. python Selenium启动chromedriver
  13. Floyd-Warshall算法过程中矩阵计算方法—十字交叉法
  14. 2021年金属非金属矿山(露天矿山)安全管理人员考试报名及金属非金属矿山(露天矿山)安全管理人员证考试
  15. 多年后再回头看那海市蜃楼
  16. SAS的基本使用介绍2(变量的输入输出格式)
  17. 英雄榜 此服务器当前正在维护中,《天下3》2021年全新山海经版本“再战黄泉”震撼登场!——网易《天下3》官方网站...
  18. 破解Excel的宏密码
  19. 中国鹊桥,国际交友网
  20. C语言第二课:项目的创建、基本结构与两个常见问题的处理

热门文章

  1. S3C6410处理器介绍
  2. 从Nand Flash启动U-BOOT的基本原理
  3. oracle无效的存储过程怎么解决,oracle数据库 编译无效存储过程
  4. 菜鸟学习笔记:Java基础篇3(面向对象思想、程序执行过程内存分析、面向对象重要概念)
  5. RabbitMQ的简单示例
  6. Spring中PropertyPlaceholderConfigurer的使用
  7. 总结对象的两个方法valueof和tostring
  8. python-24: re 模块 之二 re方法及反斜杠
  9. ntent action大全
  10. (译)C#/.NET中的委托与事件