时间测量
Kotlin 标准库提供了在不同单位下计算和测量时间的工具。 准确的时间测量对于以下活动非常重要:
- 管理线程或进程
- 收集统计信息
- 检测超时
- 调试
默认情况下,时间是使用单调时间源测量的,但也可以配置其他时间源。 有关更多信息,请参阅创建时间源。
计算时长
为了表示一段时间,标准库提供了 Duration 类。Duration 可以使用 DurationUnit 枚举类中的以下单位来表示:
NANOSECONDSMICROSECONDSMILLISECONDSSECONDSMINUTESHOURSDAYS
Duration 可以是正数、负数、零、正无穷大或负无穷大。
创建时长
要创建 Duration,请使用为 Int、Long 和 Double 类型提供的扩展属性:nanoseconds、microseconds、milliseconds、seconds、minutes、hours 和 days。
“天”是指 24 小时的周期,而非日历天。
例如:
import kotlin.time.*
import kotlin.time.Duration.Companion.nanoseconds
import kotlin.time.Duration.Companion.milliseconds
import kotlin.time.Duration.Companion.seconds
import kotlin.time.Duration.Companion.minutes
import kotlin.time.Duration.Companion.days
fun main() {
val fiveHundredMilliseconds: Duration = 500.milliseconds
val zeroSeconds: Duration = 0.seconds
val tenMinutes: Duration = 10.minutes
val negativeNanosecond: Duration = (-1).nanoseconds
val infiniteDays: Duration = Double.POSITIVE_INFINITY.days
val negativeInfiniteDays: Duration = Double.NEGATIVE_INFINITY.days
println(fiveHundredMilliseconds) // 500ms
println(zeroSeconds) // 0s
println(tenMinutes) // 10m
println(negativeNanosecond) // -1ns
println(infiniteDays) // Infinity
println(negativeInfiniteDays) // -Infinity
}您还可以对 Duration 对象执行基本的算术运算:
import kotlin.time.*
import kotlin.time.Duration.Companion.seconds
fun main() {
val fiveSeconds: Duration = 5.seconds
val thirtySeconds: Duration = 30.seconds
println(fiveSeconds + thirtySeconds)
// 35s
println(thirtySeconds - fiveSeconds)
// 25s
println(fiveSeconds * 2)
// 10s
println(thirtySeconds / 2)
// 15s
println(thirtySeconds / fiveSeconds)
// 6.0
println(-thirtySeconds)
// -30s
println((-thirtySeconds).absoluteValue)
// 30s
}获取字符串表示形式
获取 Duration 的字符串表示形式非常有用,这样您就可以对其进行输出、序列化、传输或存储。
要获取字符串表示形式,请使用 .toString() 函数。默认情况下,时间会使用存在的每个单位进行报告。例如:1h 0m 45.677s 或 -(6d 5h 5m 28.284s)。
要配置输出,请使用 .toString() 函数,并将所需的 DurationUnit 和小数位数作为函数形参:
import kotlin.time.Duration
import kotlin.time.Duration.Companion.milliseconds
import kotlin.time.DurationUnit
fun main() {
// 以秒为单位输出,保留 2 位小数
println(5887.milliseconds.toString(DurationUnit.SECONDS, 2))
// 5.89s
}要获取符合 ISO-8601 标准的字符串,请使用 toIsoString() 函数:
import kotlin.time.Duration.Companion.seconds
fun main() {
println(86420.seconds.toIsoString()) // PT24H0M20S
}转换时长
要将 Duration 转换为不同的 DurationUnit,请使用以下属性:
inWholeNanosecondsinWholeMicrosecondsinWholeSecondsinWholeMinutesinWholeHoursinWholeDays
例如:
import kotlin.time.Duration
import kotlin.time.Duration.Companion.minutes
fun main() {
val thirtyMinutes: Duration = 30.minutes
println(thirtyMinutes.inWholeSeconds)
// 1800
}或者,您可以在以下扩展函数中将所需的 DurationUnit 用作函数形参:
.toInt().toDouble().toLong()
例如:
import kotlin.time.Duration.Companion.seconds
import kotlin.time.DurationUnit
fun main() {
println(270.seconds.toDouble(DurationUnit.MINUTES))
// 4.5
}比较时长
要检查 Duration 对象是否相等,请使用相等运算符 (==):
import kotlin.time.Duration
import kotlin.time.Duration.Companion.hours
import kotlin.time.Duration.Companion.minutes
fun main() {
val thirtyMinutes: Duration = 30.minutes
val halfHour: Duration = 0.5.hours
println(thirtyMinutes == halfHour)
// true
}要比较 Duration 对象,请使用比较运算符 (<、>):
import kotlin.time.Duration.Companion.microseconds
import kotlin.time.Duration.Companion.nanoseconds
fun main() {
println(3000.microseconds < 25000.nanoseconds)
// false
}将时长拆分为组件
要将 Duration 拆分为其时间组件并执行进一步操作,请使用 toComponents() 函数的重载版本。将所需的操作作为函数或 lambda 表达式作为函数形参添加。
例如:
import kotlin.time.Duration
import kotlin.time.Duration.Companion.minutes
fun main() {
val thirtyMinutes: Duration = 30.minutes
println(thirtyMinutes.toComponents { hours, minutes, _, _ -> "${hours}h:${minutes}m" })
// 0h:30m
}在此示例中,lambda 表达式将 hours 和 minutes 作为函数形参,并对未使用的 seconds 和 nanoseconds 形参使用下划线 (_)。该表达式使用字符串模板返回一个串联字符串,以获取所需的 hours 和 minutes 输出格式。
测量时间
为了跟踪时间的流逝,标准库提供了工具,使您可以轻松地:
- 以所需的单位测量执行某些代码所需的时间。
- 标记某个时刻。
- 比较并减去两个时刻。
- 检查自某个特定时刻以来已经过去了多少时间。
- 检查当前时间是否已超过某个特定时刻。
测量代码执行时间
要测量执行一段代码块所需的时间,请使用 measureTime 内联函数:
import kotlin.time.measureTime
fun main() {
val timeTaken = measureTime {
Thread.sleep(100)
}
println(timeTaken) // 例如 103 ms
}要测量执行一段代码块所需的时间并返回该代码块的值,请使用内联函数 measureTimedValue。
例如:
import kotlin.time.measureTimedValue
fun main() {
val (value, timeTaken) = measureTimedValue {
Thread.sleep(100)
42
}
println(value) // 42
println(timeTaken) // 例如 103 ms
}默认情况下,这两个函数都使用单调时间源。
标记时刻
要标记特定的时刻,请使用 TimeSource 接口和 markNow() 函数来创建一个 TimeMark:
import kotlin.time.*
fun main() {
val timeSource = TimeSource.Monotonic
val mark = timeSource.markNow()
}测量时间差
要测量来自同一时间源的 TimeMark 对象之间的差异,请使用减法运算符 (-)。
要比较来自同一时间源的 TimeMark 对象,请使用比较运算符 (<、>)。
例如:
import kotlin.time.*
fun main() {
val timeSource = TimeSource.Monotonic
val mark1 = timeSource.markNow()
Thread.sleep(500) // 休眠 0.5 秒。
val mark2 = timeSource.markNow()
repeat(4) { n ->
val mark3 = timeSource.markNow()
val elapsed1 = mark3 - mark1
val elapsed2 = mark3 - mark2
println("测量 1.${n + 1}: elapsed1=$elapsed1, elapsed2=$elapsed2, diff=${elapsed1 - elapsed2}")
}
println(mark2 > mark1) // 结果为 true,因为 mark2 是在 mark1 之后捕获的。
// true
}要检查截止时间是否已过或是否已达到超时,请使用 hasPassedNow() 和 hasNotPassedNow() 扩展函数:
import kotlin.time.*
import kotlin.time.Duration.Companion.seconds
fun main() {
val timeSource = TimeSource.Monotonic
val mark1 = timeSource.markNow()
val fiveSeconds: Duration = 5.seconds
val mark2 = mark1 + fiveSeconds
// 还没到 5 秒
println(mark2.hasPassedNow())
// false
// 等待 6 秒
Thread.sleep(6000)
println(mark2.hasPassedNow())
// true
}时间源
默认情况下,时间是使用单调时间源测量的。单调时间源仅向前移动,不受时区等变化的影响。单调时间的一种替代方案是经过的实时时间,也称为挂钟时间。经过的实时时间是相对于另一个时间点测量的。
各平台的默认时间源
下表说明了每个平台的默认单调时间源:
| 平台 | 来源 |
|---|---|
| Kotlin/JVM | System.nanoTime() |
| Kotlin/JS (Node.js) | process.hrtime() |
| Kotlin/JS (浏览器) | window.performance.now() 或 Date.now() |
| Kotlin/Native | std::chrono::high_resolution_clock 或 std::chrono::steady_clock |
创建时间源
在某些情况下,您可能希望使用不同的时间源。例如在 Android 中,System.nanoTime() 仅在设备处于活动状态时计时。当设备进入深度休眠时,它会失去对时间的跟踪。为了在设备处于深度休眠时仍能跟踪时间,您可以创建一个使用 SystemClock.elapsedRealtimeNanos() 的时间源:
object RealtimeMonotonicTimeSource : AbstractLongTimeSource(DurationUnit.NANOSECONDS) {
override fun read(): Long = SystemClock.elapsedRealtimeNanos()
}然后,您可以使用该时间源进行时间测量:
fun main() {
val elapsed: Duration = RealtimeMonotonicTimeSource.measureTime {
Thread.sleep(100)
}
println(elapsed) // 例如 103 ms
}有关 kotlin.time 软件包的更多信息,请参阅我们的标准库 API 参考。
