避免误差!Android 中正确计算时间差的方式
在 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 + " 毫秒");
在这个例子中,startTime
和 endTime
都是使用 SystemClock.elapsedRealtime()
获取的,即便系统时间发生调整,duration
也会是准确的操作耗时。
SystemClock
中的其他计时方法
SystemClock
类中还提供了其他几种有用的计时方法:
-
SystemClock.uptimeMillis()
:返回设备自启动以来的时间,但不包括设备休眠的时间。适合用于计时操作,不受系统时间调整影响,但会忽略休眠状态。 -
SystemClock.currentThreadTimeMillis()
:返回当前线程的 CPU 时间,不包括其他线程的耗时。这在分析特定线程的执行时间时非常有用。
总结
在 Android 开发中,SystemClock.elapsedRealtime()
是计算时间差异的最佳选择,尤其是当计时过程可能跨越系统待机或涉及高精度的时间间隔计算时。它的单调递增特性和独立于系统时间调整的特点,确保了计时的连续性和稳定性。而 System.currentTimeMillis()
更适合用来获取绝对的系统时间,例如记录日志时间、事件的发生时间等。
选择合适的计时方法,不仅可以避免时间误差带来的影响,还能提升应用性能,确保计时功能的可靠性。希望本文能帮助你在实际开发中理解和应用这些计时方法,写出更高效、准确的 Android 应用!
相关文章:
避免误差!Android 中正确计算时间差的方式
在 Android 开发中,计时和计算时间差异是非常常见的需求,比如记录事件发生的间隔、统计应用启动时间、测量网络请求的响应时间等。在实现这些功能时,我们通常需要一个可靠的时间源来确保计时的准确性。那么为什么 Android 推荐使用 SystemClo…...

unity3d————Resources异步加载
知识点一:Resources异步加载是什么? 在Unity中,资源加载可以分为同步加载和异步加载两种方式。同步加载会在主线程中直接进行,如果加载的资源过大,可能会导致程序卡顿,因为从硬盘读取数据到内存并进行处理…...

YOLOv11改进,YOLOv11添加GnConv递归门控卷积,二次创新C3k2结构
摘要 视觉 Transformer 在多种任务中取得了显著的成功,这得益于基于点积自注意力的新空间建模机制。视觉 Transformer 中的关键因素——即输入自适应、长距离和高阶空间交互——也可以通过卷积框架高效实现。作者提出了递归门控卷积(Recursive Gated Convolution,简称 gnCo…...
如何选择国产化CMS来建设政务网站?
在介绍CMS之前,我们先了解国家为什么要网站为什么要完成国产化改造? 1、信创国产化网站建站响应了国家的信息安全战略,支持自主可控的信息技术产业的发展,减少对进口软硬件的依赖,保障国家信息安全。 2、国产替代&…...

C/C++语言基础--initializer_list表达式、tuple元组、pair对组简介
本专栏目的 更新C/C的基础语法,包括C的一些新特性 前言 initializer_list表达式、tuple元组、pair对组再C日常还是比较常用的,尤其是对组在刷算法还是挺好用的,这里做一个简介;这三个语法结合C17的结构化绑定会更好用ÿ…...

paddle表格识别数据制作
数据格式 其中主要数据有两个一个表格结构的检测框,一个是tokens,注意的地方是 1、只能使用双引号,单引号不行 2、使用带引号的地方是tokens里面 "<tr>", "<td", " colspan2", ">",&quo…...
python selenium库的使用:通过兴趣点获取坐标
通过兴趣点获取坐标 from selenium import webdriver from selenium.webdriver.common.by import By from selenium.webdriver.common.keys import Keys from selenium.common.exceptions import TimeoutException# 保存Cookies到文件(可选) import pi…...

如何优化Kafka消费者的性能
要优化 Kafka 消费者性能,你可以考虑以下策略: 并行消费:通过增加消费者组中的消费者数量来并行处理更多的消息,从而提升消费速度。 批量消费:配置 fetch.min.bytes 和 fetch.max.wait.ms 参数来控制批量消费的大小和…...
机器学习 决策树
决策树-分类 1 概念 1、决策节点通过条件判断而进行分支选择的节点。如:将某个样本中的属性值(特征值)与决策节点上的值进行比较,从而判断它的流向。 2、叶子节点没有子节点的节点,表示最终的决策结果。 3、决策树的深度所有节点的最大层…...
效益登记册效益管理计划
效益登记册 benefit Register效益管理计划效益登记册汇集并列出项目集计划的效益,用于在项目集的整个持续时间内测量和沟通效益的交付。在效益识别阶段,效益登记册根据项目集商业论证、组织战略计划和其他相关项目集自标而编制。随后,登记册由…...
Go语言的零值可用性:优势与限制
Go语言以其简洁和高效的设计理念而著称,其中之一便是“零值可用”的特性。这一特性使得许多类型在未显式初始化时即可直接安全地使用,大大简化了代码的初始化过程。然而,并非所有类型都支持零值可用,且在使用时也存在一定的限制。…...

【自用】0-1背包问题与完全背包问题的Java实现
引言 背包问题是计算机科学领域的一个经典优化问题,分为多种类型,其中最常见的是0-1背包问题和完全背包问题。这两种问题的核心在于如何在有限的空间内最大化收益,但它们之间存在一些关键的区别:0-1背包问题允许每个物品只能选择…...

HTML5实现俄罗斯方块小游戏
文章目录 1.设计来源1.1 主界面1.2 皮肤风格1.2 游戏中界面1.3 游戏结束界面 2.效果和源码2.1 动态效果2.2 源代码 源码下载 作者:xcLeigh 文章地址:https://blog.csdn.net/weixin_43151418/article/details/143788449 HTML5实现俄罗斯方块小游戏&#x…...
Mybatis官方生成器使用示例
在这篇文章中,我们将通过实际代码示例来说明如何使用 MyBatis Generator (MBG) 来自动化生成 MyBatis 项目所需的实体类、Mapper 接口和 Mapper XML 文件。我们将使用一个 Maven 插件来执行代码生成,并提供详细的配置和解释。 1. MyBatis Generator 简介…...

演员王子辰—专注革命题材 《前行者》后再出发
2021年10月22日在北京卫视播出的由张鲁一、聂远等人主演的电视剧《前行者》,讲述了在二十世纪三十年代初,因叛徒出卖,我上海地下党组织遭到严重破坏,革命事业陷入一片白色恐怖之中。我党情报员马天目刚从法国归来,临危…...
Spring Boot基础教学:创建第一个Spring Boot项目
使用Spring Initializr生成项目 Spring Initializr是一个在线工具,用于快速生成Spring Boot项目的基本结构。以下是使用Spring Initializr创建项目的步骤: 步骤1:访问Spring Initializr 打开网址 start.spring.io。 步骤2:选择…...

基于SpringBoot+Vue实现校园多媒体信息共享平台
作者简介:Java领域优质创作者、CSDN博客专家 、CSDN内容合伙人、掘金特邀作者、阿里云博客专家、51CTO特邀作者、多年架构师设计经验、多年校企合作经验,被多个学校常年聘为校外企业导师,指导学生毕业设计并参与学生毕业答辩指导,…...

WebRTC API分析
主题 本文详细描述常用的webrtc api 媒体协商类 myPeerConnection.createOffer([options]); var options { offerToReceiveAudio: true, // 告诉另一端,你是否想接收音频,默认true offerToReceiveVideo: true, // 告诉另一端&a…...

ArkTS学习笔记:ArkTS起步
ArkTS是HarmonyOS的主力应用开发语言,基于TypeScript扩展,强化了静态检查和分析,旨在提升程序稳定性和性能。它采用静态类型,禁止运行时改变对象布局,并对UI开发框架能力进行扩展,支持声明式UI描述和自定义…...

spring-gateway网关聚合swagger实现多个服务接口切换
前提条件 微服务已经集成了swagger,并且注册进了nacos。 gateway配置 package com.zmy.springcloud.config;import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springfra…...
Linux链表操作全解析
Linux C语言链表深度解析与实战技巧 一、链表基础概念与内核链表优势1.1 为什么使用链表?1.2 Linux 内核链表与用户态链表的区别 二、内核链表结构与宏解析常用宏/函数 三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势5.1 插入效率5.2 安全…...
Leetcode 3576. Transform Array to All Equal Elements
Leetcode 3576. Transform Array to All Equal Elements 1. 解题思路2. 代码实现 题目链接:3576. Transform Array to All Equal Elements 1. 解题思路 这一题思路上就是分别考察一下是否能将其转化为全1或者全-1数组即可。 至于每一种情况是否可以达到…...
React hook之useRef
React useRef 详解 useRef 是 React 提供的一个 Hook,用于在函数组件中创建可变的引用对象。它在 React 开发中有多种重要用途,下面我将全面详细地介绍它的特性和用法。 基本概念 1. 创建 ref const refContainer useRef(initialValue);initialValu…...

Android15默认授权浮窗权限
我们经常有那种需求,客户需要定制的apk集成在ROM中,并且默认授予其【显示在其他应用的上层】权限,也就是我们常说的浮窗权限,那么我们就可以通过以下方法在wms、ams等系统服务的systemReady()方法中调用即可实现预置应用默认授权浮…...

基于IDIG-GAN的小样本电机轴承故障诊断
目录 🔍 核心问题 一、IDIG-GAN模型原理 1. 整体架构 2. 核心创新点 (1) 梯度归一化(Gradient Normalization) (2) 判别器梯度间隙正则化(Discriminator Gradient Gap Regularization) (3) 自注意力机制(Self-Attention) 3. 完整损失函数 二…...

【C++进阶篇】智能指针
C内存管理终极指南:智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...
掌握 HTTP 请求:理解 cURL GET 语法
cURL 是一个强大的命令行工具,用于发送 HTTP 请求和与 Web 服务器交互。在 Web 开发和测试中,cURL 经常用于发送 GET 请求来获取服务器资源。本文将详细介绍 cURL GET 请求的语法和使用方法。 一、cURL 基本概念 cURL 是 "Client URL" 的缩写…...

QT开发技术【ffmpeg + QAudioOutput】音乐播放器
一、 介绍 使用ffmpeg 4.2.2 在数字化浪潮席卷全球的当下,音视频内容犹如璀璨繁星,点亮了人们的生活与工作。从短视频平台上令人捧腹的搞笑视频,到在线课堂中知识渊博的专家授课,再到影视平台上扣人心弦的高清大片,音…...

数据结构:泰勒展开式:霍纳法则(Horner‘s Rule)
目录 🔍 若用递归计算每一项,会发生什么? Horners Rule(霍纳法则) 第一步:我们从最原始的泰勒公式出发 第二步:从形式上重新观察展开式 🌟 第三步:引出霍纳法则&…...
跨平台商品数据接口的标准化与规范化发展路径:淘宝京东拼多多的最新实践
在电商行业蓬勃发展的当下,多平台运营已成为众多商家的必然选择。然而,不同电商平台在商品数据接口方面存在差异,导致商家在跨平台运营时面临诸多挑战,如数据对接困难、运营效率低下、用户体验不一致等。跨平台商品数据接口的标准…...