代码改变世界

Android 中获取稳定时间的方法 - 指南

2025-09-22 20:10  tlnshuju  阅读(13)  评论(0)    收藏  举报

在Android中,获取不会因系统时间设置改变而跳变的时间通常需要利用基于设备启动时间的单调时钟。下面我将展示几种实现方式。

实现思路

1. 运用SystemClock.elapsedRealtime() - 基于设备启动时间,不受系统时间设置影响
2. 使用SystemClock.uptimeMillis() - 同样基于设备启动时间,但不包括深度睡眠时间
3. 使用System.nanoTime() - 高精度单调时间

示例代码实现

```xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="20dp"
android:orientation="vertical"
android:gravity="center">

<TextView
android:id="@+id/titleTextView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="稳定时间获取示例"
android:textSize="20sp"
android:textStyle="bold"
android:layout_marginBottom="30dp"/>

<Button
android:id="@+id/elapsedRealtimeButton"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="获取elapsedRealtime"
android:layout_marginBottom="10dp"/>

<Button
android:id="@+id/uptimeMillisButton"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="获取uptimeMillis"
android:layout_marginBottom="10dp"/>

<Button
android:id="@+id/nanoTimeButton"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="获取nanoTime"
android:layout_marginBottom="30dp"/>

<TextView
android:id="@+id/resultTextView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="结果将显示在这里"
android:textSize="16sp"
android:gravity="center"
android:padding="10dp"
android:background="#f0f0f0"/>

</LinearLayout>
```

```java
import android.os.Bundle;
import android.os.SystemClock;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import androidx.appcompat.app.AppCompatActivity;

public class MainActivity extends AppCompatActivity {

private TextView resultTextView;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);

resultTextView = findViewById(R.id.resultTextView);
Button elapsedRealtimeButton = findViewById(R.id.elapsedRealtimeButton);
Button uptimeMillisButton = findViewById(R.id.uptimeMillisButton);
Button nanoTimeButton = findViewById(R.id.nanoTimeButton);

elapsedRealtimeButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
long elapsedTime = SystemClock.elapsedRealtime();
resultTextView.setText("elapsedRealtime: " + elapsedTime + "ms\n" +
"格式化为时间: " + formatTime(elapsedTime));
}
});

uptimeMillisButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
long uptime = SystemClock.uptimeMillis();
resultTextView.setText("uptimeMillis: " + uptime + "ms\n" +
"格式化为时间: " + formatTime(uptime));
}
});

nanoTimeButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
long nanoTime = System.nanoTime();
resultTextView.setText("nanoTime: " + nanoTime + "ns\n" +
"转换为毫秒: " + (nanoTime / 1_000_000) + "ms");
}
});
}

private String formatTime(long milliseconds) {
long seconds = milliseconds / 1000;
long minutes = seconds / 60;
long hours = minutes / 60;
long days = hours / 24;

return String.format("%d天 %02d:%02d:%02d.%03d",
days, hours % 24, minutes % 60, seconds % 60, milliseconds % 1000);
}
}
```

关键说明

1. SystemClock.elapsedRealtime():
· 自设备启动以来经过的时间,包括深度睡眠时间
· 最适合测量时间间隔,不受系统时间设置影响
2. SystemClock.uptimeMillis():
· 自设备启动以来经过的时间,不包括深度睡眠时间
· 适合应该忽略设备睡眠时间的场景
3. System.nanoTime():
· 提供纳秒级精度的时间
· 通常用于性能测量和高精度计时

这些技巧都提供了单调递增的时间值,不会因为用户更改系统时间或自动时间同步而跳变。

使用场景建议

· 测量时间间隔或耗时操作:使用elapsedRealtime()或uptimeMillis()
· 高精度性能测量:利用nanoTime()
· 需跨设备重启保持时间跟踪:需要将时间持久化存储并与elapsedRealtime()结合使用

注意:如果必须跨设备重启保持时间跟踪,允许将设备启动时间与elapsedRealtime()结合使用,但这必须额外的实现。