避免误差!Android 中正确计算时间差的方式

news2024/11/17 20:55:08

在 Android 开发中,计时和计算时间差异是非常常见的需求,比如记录事件发生的间隔、统计应用启动时间、测量网络请求的响应时间等。在实现这些功能时,我们通常需要一个可靠的时间源来确保计时的准确性。那么为什么 Android 推荐使用 SystemClock.elapsedRealtime() 来计算时间差异,而不建议使用 System.currentTimeMillis() 呢?本文将详细探讨这个问题。

System.currentTimeMillis()SystemClock.elapsedRealtime() 的区别

在 Android 中,System.currentTimeMillis()SystemClock.elapsedRealtime() 都能获取时间,但两者之间存在明显差异:

  • System.currentTimeMillis() :返回的是当前的“系统时间”,从 1970 年 1 月 1 日 UTC 以来的毫秒数。这个时间可以通过网络同步、用户手动更改等方式调整,时间的准确性和连续性不一定可靠。
  • SystemClock.elapsedRealtime() :返回的是设备自上次启动以来的毫秒数。该计时器是 单调递增的,也就是说,即使系统时间被调整,或者设备进入了待机模式,这个计时器也会保持更新,始终提供可靠的时间差。

为什么选择 SystemClock.elapsedRealtime() 计算时间差异?

在 Android 中,更推荐使用 SystemClock.elapsedRealtime() 来计算时间差异,主要基于以下几点原因:

1. 时间连续性和稳定性

System.currentTimeMillis() 可能会因为系统时间的调整而出现时间跳变或倒退,比如用户手动更改时间、网络时间同步等。这会导致时间间隔的计算结果不准确。

相反,SystemClock.elapsedRealtime() 是设备启动后的时间流逝计时,不会受到系统时间调整的影响。这就保证了在任何情况下,使用 SystemClock.elapsedRealtime() 来计算的时间差值都是稳定和连续的。因此,如果你希望获得可靠的时间差,那么 elapsedRealtime() 是更好的选择。

2. 适合计算相对时间差

在 Android 开发中,SystemClock.elapsedRealtime() 更适合用于计算两个时间点之间的相对时间差。因为它仅仅表示系统启动后的流逝时间,不关心系统当前的时间和日期,也不会受到系统时间变化的影响。这种设计非常适合用于统计两个事件的时间差。

例如,你可以用 SystemClock.elapsedRealtime() 来测量应用启动时间、功能执行耗时,或统计用户打开某个页面的时间差。只要记录开始和结束时间的 elapsedRealtime() 值,差值即为准确的流逝时间。

3. 设备休眠不影响计时

SystemClock.elapsedRealtime() 会考虑设备的休眠状态,即使设备进入待机状态,elapsedRealtime() 计时器也会继续更新。这意味着即便设备休眠,计时结果也不会中断或失效。这对许多需要跨越设备待机的计时操作来说非常重要,比如统计用户停留在某个页面的总时间。

相较之下,System.currentTimeMillis() 无法保证这一点,因为系统时间的变化会干扰计时效果。对于需要高精度的计时需求,elapsedRealtime() 显然更加适用。

使用场景对比

下面列举一些常见的场景,分别说明适合 SystemClock.elapsedRealtime()System.currentTimeMillis() 的情况:

使用场景推荐方法原因
计算两个事件的时间间隔SystemClock.elapsedRealtime()保证计时的准确性和连续性,不受系统时间调整影响
记录日志时间戳System.currentTimeMillis()需要绝对的时间信息来标记事件的发生时间
网络请求的响应时间SystemClock.elapsedRealtime()计时不受系统时间调整影响,确保响应时间的准确性
应用启动时间或活动的运行时长SystemClock.elapsedRealtime()提供稳定的相对时间,适合统计启动或运行耗时
获取当前日期和时间System.currentTimeMillis()返回系统时间,以获取绝对的日期时间(如显示给用户或存储)

示例代码:如何用 SystemClock.elapsedRealtime() 计算时间差

假设我们要测量某个操作的执行耗时,以下是使用 SystemClock.elapsedRealtime() 的实现方式:

 // 记录操作开始的时间戳
 long startTime = SystemClock.elapsedRealtime();
 
 // 执行操作
 performOperation();
 
 // 记录操作结束的时间戳
 long endTime = SystemClock.elapsedRealtime();
 
 // 计算操作耗时
 long duration = endTime - startTime;
 Log.d("Timing", "操作耗时:" + duration + " 毫秒");

在这个例子中,startTimeendTime 都是使用 SystemClock.elapsedRealtime() 获取的,即便系统时间发生调整,duration 也会是准确的操作耗时。

SystemClock 中的其他计时方法

SystemClock 类中还提供了其他几种有用的计时方法:

  • SystemClock.uptimeMillis() :返回设备自启动以来的时间,但不包括设备休眠的时间。适合用于计时操作,不受系统时间调整影响,但会忽略休眠状态。
  • SystemClock.currentThreadTimeMillis() :返回当前线程的 CPU 时间,不包括其他线程的耗时。这在分析特定线程的执行时间时非常有用。

总结

在 Android 开发中,SystemClock.elapsedRealtime() 是计算时间差异的最佳选择,尤其是当计时过程可能跨越系统待机或涉及高精度的时间间隔计算时。它的单调递增特性和独立于系统时间调整的特点,确保了计时的连续性和稳定性。而 System.currentTimeMillis() 更适合用来获取绝对的系统时间,例如记录日志时间、事件的发生时间等。

选择合适的计时方法,不仅可以避免时间误差带来的影响,还能提升应用性能,确保计时功能的可靠性。希望本文能帮助你在实际开发中理解和应用这些计时方法,写出更高效、准确的 Android 应用!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2242395.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

TensorFlow 2.0 环境配置

官方文档:CUDA Installation Guide for Windows 官方文档有坑,windows的安装指南直接复制了linux的指南内容:忽略这些离谱的信息即可。 可以从官方文档知悉,cuda依赖特定版本的C编译器。但是我懒得为了一个编译器就下载整个visua…

【计算机网络】【传输层】【习题】

计算机网络-传输层-习题 文章目录 10. 图 5-29 给出了 TCP 连接建立的三次握手与连接释放的四次握手过程。根据 TCP 协议的工作原理,请填写图 5-29 中 ①~⑧ 位置的序号值。答案技巧 注:本文基于《计算机网络》(第5版)吴功宜、吴英…

HarmonyOS本地存储-Preferences(用户首选项)的使用

一,用户首选项简述 ohos.data.preferences (用户首选项) 用户首选项为应用提供Key-Value键值型的数据处理能力,支持应用持久化轻量级数据,并对其修改和查询。 数据存储形式为键值对,键的类型为字符串型,值的存储数据…

基于Java Web 的家乡特色菜推荐系统

博主介绍:✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取源码联系🍅 👇🏻 精彩专栏推荐订阅👇…

【售前方案】工业园区整体解决方案,智慧园区方案,智慧城市方案,智慧各类信息化方案(ppt原件)

基于云计算、物联网、移动通信计算的智慧园区集中运营管理平台是一个高度集成化、智能化的管理系统,它利用先进的技术手段对园区进行全方位的监控和管理。 软件资料清单列表部分文档清单:工作安排任务书,可行性分析报告,立项申请审…

Vue2+ElementUI:用计算属性实现搜索框功能

前言: 本文代码使用vue2element UI。 输入框搜索的功能,可以在前端通过计算属性过滤实现,也可以调用后端写好的接口。本文介绍的是通过计算属性对表格数据实时过滤,后附完整代码,代码中提供的是死数据,可…

CSS实现炫酷的水波纹效果

炫酷的水波纹效果 看好了&#xff0c;下面是我最后的波纹了 实现代码 HTML&#xff1a; <div class"container"></div> <script src"https://cdn.bootcdn.net/ajax/libs/jquery/3.7.1/jquery.js"></script>CSS: body{height: 1…

【会话文本nlp】对话文本解析库pyconverse使用教程版本报错、模型下载等问题解决超参数调试

前言&#xff1a; 此篇博客用于记录调用pyconverse库解析对话文本时遇到的问题与解决思路&#xff0c;以供大家参考。 文章目录 pycoverse介绍代码github链接问题解决1 [cannot import name ‘cached_download‘ from ‘huggingface_hub‘ 问题解决](https://blog.csdn.net/wei…

单元测试时报错找不到@SpringBootConfiguration

找到问题出现原因&#xff1a; 错误表示 Spring Boot 在运行测试时无法找到 SpringBootConfiguration 注解。 通常&#xff0c;SpringBootTest注解用于加载 Spring Boot 应用上下文&#xff0c;但它需要找到一个带有SpringBootConfiguration&#xff08;或者Configuration&am…

【数据结构】11.哈夫曼树哈夫曼编码

一、哈夫曼树的基本概念 哈夫曼&#xff08;Huffman&#xff09;树又称最优树&#xff0c;是一类带权路径长度最短的树&#xff0c;在实际中有广泛的用途。 路径&#xff1a; 从树中一个节点到另一个节点之间的分支构成这两个节点之间的路径。路径长度&#xff1a; 路径上的分…

AntD表单自定义组件

前言 表单可以说是前端最常见的一种组件&#xff0c;特别是在进行搜索的时候使用的最频繁&#xff0c;自定义表单组件&#xff0c;丰富了搜索框的类型&#xff0c;使数据展现的更灵活 内容讲解 1、官方介绍 AntD-Formhttps://ant.design/components/form-cn#form-demo-cust…

day-83 最少翻转次数使二进制矩阵回文 II

思路 关键在于1的个数要为4的倍数&#xff0c;首先镜像的四个位置肯定一定为4的倍数&#xff0c;如果行和列为奇数则需要单独考虑&#xff0c;如果行和列皆为奇数&#xff0c;那么中心的那个数一定为0 解题过程 再单独考虑如果行和列为奇数&#xff0c;具体参考灵神。如果diff…

Gitcode文件历史记录查看和还原

文件历史记录 文件历史记录用于记录代码文件的更改历史&#xff0c;它允许用户查看文件的不同版本&#xff0c;了解每个版本的修改内容、作者和提交消息。这对于跟踪文件演进、恢复错误更改、审查代码以及了解项目进展都非常有用。 文件历史记录功能提供了以下核心功能&#…

WebSocket协议在Java中的整合

1. 常见的消息推送方式 2.WebSocket API 3.基于WebSocket的实战&#xff08;实时聊天室&#xff09; 这里以解析后端代码为主&#xff0c;前端不作为重点&#xff0c;若想复现项目&#xff0c;请从作者的仓库中拉取代码 WebSocket-chatRoom: 基于WebSocket协议实现一个简单的…

http自动发送请求工具(自动化测试http请求)

点击下载《http自动发送请求工具(自动化测试http请求)》 前言 在现代软件开发过程中&#xff0c;HTTP 请求的自动化测试是确保应用程序稳定性和可靠性的关键环节。为了满足这一需求&#xff0c;我开发了一款功能强大且易于使用的自动化 HTTP 请求发送工具。该工具基于 C# 开发…

【小白可懂】微信小程序---课表渲染

结果展示&#xff1a;&#xff08;代码在最后&#xff09; WeChat_20241116174431 项目简介 在数字化校园建设的大背景下&#xff0c;为了更好地服务于在校师生&#xff0c;我们开发了一款基于微信小程序的课表管理系统。该系统采用了现代化的前端技术和优雅的设计风格&#x…

WinDefender Weaker

PPL Windows Vista / Server 2008引入 了受保护进程的概念&#xff0c;其目的不是保护您的数据或凭据。其最初目标是保护媒体内容并符合DRM &#xff08;数字版权管理&#xff09;要求。Microsoft开发了此机制&#xff0c;以便您的媒体播放器可以读取例如蓝光&#xff0c;同时…

LabVIEW前面板最大化显示与像素偏差分析 有源程序附件

LabVIEW前面板最大化显示与像素偏差分析 有源程序附件 LabVIEW前面板最大化显示与像素偏差分析 有源程序附件 - 北京瀚文网星科技有限公司 这个VI用于将LabVIEW程序的前面板最大化地显示在指定显示器上&#xff0c;实现步骤如下&#xff1a; 1. 获取所有显示器的信息 首先&…

【C++】深入理解 C++ 优先级队列、容器适配器与 deque:实现与应用解析

个人主页: 起名字真南的CSDN博客 个人专栏: 【数据结构初阶】 &#x1f4d8; 基础数据结构【C语言】 &#x1f4bb; C语言编程技巧【C】 &#x1f680; 进阶C【OJ题解】 &#x1f4dd; 题解精讲 目录 前言&#x1f4cc; 1. 优先级队列、容器适配器和 deque 概述✨1.1 什么是优…

LogViewer NLog, Log4Net, Log4j 文本日志可视化

LogViewer 下载 示例&#xff1a;NLog文本日志可视化软件&#xff0c;并且能够实时监听输出最新的日志 nlog.config 通过udp方式传输给LogViewer (udp://ip:port) <?xml version"1.0" encoding"utf-8" ?> <nlog xmlns"http://www.nlog-…