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

android–HandlersvsThreads是否有任何真正的性能影响?

关于Threadsvs.Handlers与AsyncTask的正确使用有很多问题.(如here和here)这些问题很好地解决了何时使用什么问题.我的问题更多的是关于某些类型案件的绩

关于Threads vs. Handlers与AsyncTask的正确使用有很多问题. (如here和here)

这些问题很好地解决了何时使用什么问题.我的问题更多的是关于某些类型案件的绩效影响.

作为一个例子,我经常看到其他人编写代码,他们使用Threads只是为了能够为将来安排一些代码执行.每当我看到这一点时,我本能地感觉要重构代码以使用Handler而只是发布一个延迟的runnable.

下面是一个示例,其中一个Thread用于更新某些媒体播放器的搜索栏,然后是我的方式.

我看到了很多东西:

if (positionTracker != null && positionTracker.isAlive()
&& !positionTracker.isInterrupted()) {
return;
}
positiOnTracker= new Thread(new Runnable() {
public void run() {
int currentPosition = 0;
int total = player.getDuration();
while (player != null && CurrentPosition try {
Thread.sleep(1000);
currentPosition = player.getCurrentPosition();
} catch (InterruptedException e) {
return;
} catch (Exception e) {
return;
}
if (someListener != null) {
someListener.onEvent();
}
}
}
}, "position tracker thread");
positionTracker.start();

我喜欢这样做的方式:

Runnable trackPositiOnRunnable= new Runnable() {
@Override
public void run() {
currentPosition = player.getCurrentPosition();
if (someListener != null) {
someListener.onEvent();
mHandler.postDelayed(this, 1000);
}
}
};
mHandler.post(trackPositionRunnable);

显然,我喜欢的方式是更容易阅读和更简洁.但是性能影响是什么?在性能方面,单向方法比其他方法更好吗?如果是这样,为什么?

解决方法:

每种方法都取决于您计划在Runnable中做什么,以确定它是否有用.它们之间最大的区别在于您是否计划触摸UI.在Android中,您无法触摸UI线程中的UI组件(您的媒体播放器示例正在使用原始线程破坏此规则).由于这个规则,它立即划分了每种方法所能做的和不能做的事情.这些方法之间的性能差异可以忽略不计,因为运行后台工作所花费的时间将胜过它们之间的任何差异.

处理程序通常使用另一个后台线程来执行逻辑,但它取决于构造处理程序的线程.如果Handler是在UI Thread上构建的(响应于回调onSomething),那么Runnable将在UI Thread内部运行,这使得触摸UI组件成为可能.但是,如果您在UI线程上创建了Runnables,则无法触及UI组件.在UI线程上创建的处理程序的缺点意味着您没有在后台执行这些操作,因此如果作业需要很长时间才能运行,它将锁定UI直到完成.虽然从非UI线程运行的处理程序将修复锁定UI的任何问题.他们需要做更多的工作来设置,你仍然需要争论如何安全地更新UI以响应你的后台工作(即如果你想更新UI,你仍然必须将另一个可运行的东西发回UI线程).

原始线程不会锁定UI,因为它们独立于UI线程运行,但您无法触及它们上的UI组件.这意味着您必须在UI线程上执行要更新UI的任何代码,这意味着需要编写更多代码才能让UI线程运行它.这可能非常复杂.由于使用它们的复杂性,应该真正避免原始线程.

后台任务的最常见示例是等待来自服务器的响应.大多数库都会阻塞,直到服务器发送响应,这意味着您无法在UI线程上调用它们,否则在服务器返回呼叫之前,您的用户将无法执行任何操作.它们不仅会被阻止,而且UI无法自我更新以显示微调器或以其他方式看起来活着.这最好推到后台线程.技术上处理程序和线程可以做到这一点,但处理程序必须专门构造,以便它们将使用真正的后台线程.

这是AsyncTask胜过处理程序的地方,因为它同时执行真正的后台作业和UI更新.它有一个部分用于在后台执行一些长时间运行的操作,它有一个部分用于在UI线程完成时从UI线程更新UI.它甚至还有一个可选的进度部分,因此您可以在任务运行时向UI提供任何中间进度. AsyncTask的缺点是它们必须有一个结束.继续运行以定期检查是否发生了某些事情的后台作业,睡眠和检查更多不会有助于AsyncTask模型.但是,这并不是说你不能使用Handler来定期启动AsyncTask,而只是为了讨论的完整性我提到了这一点.

最后使用原始线程并不是那么简单甚至“更好”,因为处理程序可以做很多线程可以用更少的代码做的事情.但是,处理程序在确定Runnable正在执行哪个线程时很棘手.通常它是UI线程,并且技术上将其设置为使用非UI线程是棘手的.这两个选项都受到UI更新问题的影响,因为您必须在真正的后台作业结束时执行额外的工作来运行UI作业. AsyncTask是我做背景作业的首选方法.


推荐阅读
  • 本文探讨了Java中有效停止线程的多种方法,包括使用标志位、中断机制及处理阻塞I/O操作等,旨在帮助开发者避免使用已废弃的危险方法,确保线程安全和程序稳定性。 ... [详细]
  • 深入解析mt_allocator内存分配器(二):多线程与单线程场景下的实现
    本文详细介绍了mt_allocator内存分配器在多线程和单线程环境下的实现机制。该分配器以2的幂次方字节为单位分配内存,支持灵活的配置和高效的性能。文章分为内存池特性描述、内存池实现、单线程内存池实现、内存池策略类实现及多线程内存池实现等部分,深入探讨了内存池的初始化、内存分配与回收的具体实现。 ... [详细]
  • 本文介绍了进程的基本概念及其在操作系统中的重要性,探讨了进程与程序的区别,以及如何通过多进程实现并发和并行。文章还详细讲解了Python中的multiprocessing模块,包括Process类的使用方法、进程间的同步与异步调用、阻塞与非阻塞操作,并通过实例演示了进程池的应用。 ... [详细]
  • 本文将作为我硕士论文的一部分,但鉴于其内容的独特性和趣味性,决定单独发布。文中将定义一些皮亚诺公理,并介绍如何使用这些公理进行等式替换,以证明定理。 ... [详细]
  • 深入解析轻量级数据库 SQL Server Express LocalDB
    本文详细介绍了 SQL Server Express LocalDB,这是一种轻量级的本地 T-SQL 数据库解决方案,特别适合开发环境使用。文章还探讨了 LocalDB 与其他轻量级数据库的对比,并提供了安装和连接 LocalDB 的步骤。 ... [详细]
  • 2022年4月15日的算法练习题,包括最长公共子序列和线段树的应用。 ... [详细]
  • 本文详细介绍了Oracle RMAN中的增量备份机制,重点解析了差异增量和累积增量备份的概念及其在不同Oracle版本中的实现。通过对比两种备份方式的特点,帮助读者选择合适的备份策略。 ... [详细]
  • 构建Python自助式数据查询系统
    在现代数据密集型环境中,业务团队频繁需要从数据库中提取特定信息。为了提高效率并减少IT部门的工作负担,本文探讨了一种利用Python语言实现的自助数据查询工具的设计与实现。 ... [详细]
  • 本文详细介绍了如何使用 Python 编程语言中的 Scapy 库执行 DNS 欺骗攻击,包括必要的软件安装、攻击流程及代码示例。 ... [详细]
  • 深入解析C++ Atomic编程中的内存顺序
    在多线程环境中,为了防止多个线程同时修改同一数据导致的竞争条件,通常会使用内核级同步对象,如事件、互斥锁和信号量等。然而,这些方法往往伴随着高昂的上下文切换成本。本文将探讨如何利用C++11中的原子操作和内存顺序来优化多线程编程,减少不必要的开销。 ... [详细]
  • 工作中频繁在不同Linux服务器之间切换时,频繁输入密码不仅耗时还影响效率。本文介绍如何通过设置SSH密钥认证,简化登录流程,提高工作效率。 ... [详细]
  • 本文介绍了一个基本的同步Socket程序,演示了如何实现客户端与服务器之间的简单消息传递。此外,文章还概述了Socket的基本工作流程,并计划在未来探讨同步与异步Socket的区别。 ... [详细]
  • 本文探讨了在SharePoint环境中使用BDC(Business Data Catalog)时遇到的问题及其解决策略,包括XML文件导入SSP后的不可见性问题以及与远程SQL Server 2005连接的难题。 ... [详细]
  • 深入解析Java并发之ArrayBlockingQueue
    本文详细探讨了ArrayBlockingQueue,这是一种基于数组实现的阻塞队列。ArrayBlockingQueue在初始化时需要指定容量,因此它是一个有界的阻塞队列。文章不仅介绍了其基本概念和数据结构,还深入分析了其源码实现,包括各种入队、出队、获取元素和删除元素的方法。 ... [详细]
  • Exploring issues and solutions when defining multiple Faust agents programmatically. ... [详细]
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社区 版权所有