android - Android上的循环定时器独立于系统时钟
问题描述
我正在寻找一种在 Android 上创建循环计时器的方法,该计时器独立于用户更改的系统时钟工作。根据我的研究,ScheduledExecutorService 应该是这种情况,但是当系统时间设置为过去时计时器停止触发,并在设置回现在时恢复。当系统时钟设置为将来时,此计时器会继续工作。
这是用于此的代码类型:
ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(10);
scheduler.scheduleAtFixedRate(new Runnable() {
@Override
public void run() {
Log.d(TAG, "timer hit");
}
}, 0, 2000, TimeUnit.MILLISECONDS);
然后,我尝试通过系统 Intent 监听正在更改的系统时钟,并在收到 Intent 时重新启动计时器。当系统时钟设置为未来或现在,而不是过去时,我捕捉到了 TIME_SET 意图。
我创建了一个广播接收器:
public class TimeChangedBroadcastReceiver extends BroadcastReceiver {
public static final IntentFilter intentFilter = new IntentFilter();
static {
intentFilter.addAction(Intent.ACTION_TIME_CHANGED);
intentFilter.addAction(Intent.ACTION_TIMEZONE_CHANGED);
intentFilter.addAction(Intent.ACTION_DATE_CHANGED);
}
@Override
public void onReceive(Context context, Intent intent) {
final String action = intent.getAction();
if (action == null) {
Log.d(TAG, "intent received: null");
return;
}
Log.d(TAG, "intent received: " + action);
if (action.equals(Intent.ACTION_TIME_CHANGED) ||
action.equals(Intent.ACTION_TIMEZONE_CHANGED) ||
action.equals(Intent.ACTION_DATE_CHANGED)) {
Log.d(TAG, "system time changed to " + new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
Timers.resetTimers(); //reset the instantiated timers, in another class, not included here, works correctly when it gets here
}
}
}
我将 BroadcastReceiver 添加到我的 AndroidManifest 中:
<application ...>
<receiver android:name=".TimeChangedBroadcastReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.TIME_SET"/>
<action android:name="android.intent.action.TIMEZONE_CHANGED"/>
<action android:name="android.intent.action.DATE_CHANGED"/>
</intent-filter>
</receiver>
</application>
我注册了广播接收器:
context.registerReceiver(new TimeChangedBroadcastReceiver(), TimeChangedBroadcastReceiver.intentFilter);
下面是我在运行 Android 7.1.2 的 Google Pixel 上运行 ScheduledExecutorService 和 BroadcastReceiver 时的日志输出。我将系统时钟设置为当前时间(2019),然后将年份设置为 2020,将年份设置为 2019,将年份设置为 2018,然后将年份设置为 2019。
2019-01-09 12:34:31.605 TimeChangedBroadcastReceiver registered
2019-01-09 12:34:31.606 Current time: 2019-01-09 12:34:31
2019-01-09 12:34:33.604 timer hit
2019-01-09 12:34:35.603 timer hit
2019-01-09 12:34:37.603 timer hit
2019-01-09 12:34:39.604 timer hit
2019-01-09 12:34:41.604 timer hit
2020-01-09 12:34:43.271 intent received: android.intent.action.TIME_SET
2020-01-09 12:34:43.272 system time changed to 2020-01-09 12:34:43
2020-01-09 12:34:43.273 timer hit
2020-01-09 12:34:45.273 timer hit
2020-01-09 12:34:47.273 timer hit
2019-01-09 12:34:48.812 intent received: android.intent.action.TIME_SET
2019-01-09 12:34:48.816 system time changed to 2019-01-09 12:34:48
2019-01-09 12:34:48.817 timer hit
2019-01-09 12:34:50.817 timer hit
2019-01-09 12:34:52.816 timer hit
2019-01-09 12:34:54.816 timer hit
***Here, I changed the system clock to the year 2018***
***No TIME_SET Intent is received when the system clock is changed to 2018***
***The timer stops hitting while the system clock is in 2018***
2019-01-09 12:35:02.760 intent received: android.intent.action.TIME_SET
2019-01-09 12:35:02.761 system time changed to 2019-01-09 12:35:02
2019-01-09 12:35:02.761 timer hit
2019-01-09 12:35:04.762 timer hit
2019-01-09 12:35:06.762 timer hit
如何在 Android 上创建一个循环计时器,该计时器独立于用户更改的系统时钟,或者我当前的尝试哪里出错了?
编辑:使用 handler.postDelayed() 也不起作用,行为类似于 ScheduledExecutorService
解决方案
测量时间间隔
用于SystemClock.elapsedRealTime()
测量时间间隔。
此方法返回自设备启动以来的毫秒数,与系统时间无关,包括在深度睡眠中花费的时间。
作为绝对时间,这是一个毫无意义的数字(它与世界时间和日期无关),但可以将其与以前的值进行比较以测量经过的时间。
延迟后调用函数
现在,要在一段时间后调用函数,您可以使用Handler.postDelayed
,如下所示:
final long delay = 1000; // 1 second
final Handler handler = new Handler(Looper.getMainLooper());
handler.postDelayed(
new Runnable() { public void run() {} },
delay
);
请注意,这将在主线程上运行。当它触发时,您必须自己将回调路由到后台执行器服务。
重复调用函数
使用与上面相同的代码,但handler.postDelayed()
在回调完成运行后再次调用。每个回调安排下一个。
当您想中断计时器时,只需不要重新安排回调
new Runnable() {
public void run() {
if (shouldStop) return; // don't run again if canceled after last call
doSomething();
handler.postDelayed(this, delay); // reschedule
}
};
推荐阅读
- xamarin - 如何获取其他 Android 模拟器模型以在 VS 2019 调试器中使用?
- node.js - NVM 不会在 Mac 上更改 Node 版本
- django - FOREIGN KEY 约束失败。电子邮件地址中的 Django allauth 错误
- css - 使用 Bootstrap 放置响应式 DIV
- r - 将值大于 n 的行在 R 中求和为一
- angular - 使用自定义组件时出现“未知元素”
- html - 为什么VSCODE会弄乱js文件中的jsx格式?
- reactjs - '(0, _reactRedux.connect)(mapStateToProps)(mapDispatchToProps)' 是 Object 的一个实例)
- oracle - dbms_crypto 和 Powershell
- odbc - 谓词不会下推到查询中