首页 > 解决方案 > 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

标签: android

解决方案


测量时间间隔

用于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
    }
};

推荐阅读