javascript - 为什么在Android中旋转屏幕时setIntervals不能被杀死?

标签 javascript android timer setinterval screen-rotation

各位开发者,

我正在设计一个计时器,它应该能够在屏幕旋转时保持时间,但我注意到以下代码有一个非常奇怪的行为:

function startTimer() {
    var setTime = setInterval(function () {
        console.log('increasing');
        if (isPaused)
            clearInterval(setTime);
        else {
            counter++;
            setLocalStorage("counter", counter + 1);
        }
    }, 1000);
}

function setLocalStorage(key, val) {
    if (getLocalStorageValue(key) !== null)
        removeLocalStorage(key);
    localStorage.setItem(key, val);
}

function getLocalStorageValue(key) {
    return localStorage.getItem(key);
}

function removeLocalStorage(key) {
    localStorage.removeItem(key);
}

我发现旋转屏幕后,计数器还在不停地增加,控制台甚至不打印console.log。我想 setInterval 在某个地方继续并行运行,但我找不到停止它的方法,因为我可以将 isPaused 的变量更改为 true,但计数器仍在运行没有正当理由,它会停止,直到应用程序完全关闭。你们有没有遇到过类似的事情?

此外,从 Android 方面来看,我每次都完全重新加载 HTML 页面,那么它不应该保留任何内容,或者您​​知道每次完全重新加载浏览器的方法吗?

此外,我在 Android 中添加了这段代码,但没有任何反应:

@Override
protected void onDestroy() {
    super.onDestroy();
    WebView.pauseTimers();
}

感谢您的建议或支持。

最佳答案

经过一些研究,我发现了如何解决这个问题以及发生了什么。

首先,如果您正在运行 setInterval,并且在旋转屏幕时没有在 WebView 中清除它,则原始 setInterval 将继续在后台某处运行,如果您再次旋转它并且您第二次运行计时器,它将创建另一个间隔,依此类推。

如果您希望能够修复它,则需要在每次旋转屏幕时清理页面。我是怎么做到的?

在继续之前的一个小评论,我的解决方案是用 C# 编写的,因为我在 Xamarin.Android 中构建应用程序;但是,您可以在眨眼之间将其翻译成 Java 或 Kotlin,它看起来像这样:

  1. 声明一个静态变量来检查应用程序是否是第一次恢复:

    public static bool RUN_ONCE_RESUME { get; set; } = false;
    
  2. 重写 OnResumeOnDestroy 方法如下:

    如果不是第一次,OnResume 必须重新启用计时器:

    protected override void OnResume()
    {
        base.OnResume();
    
        if (RUN_ONCE_RESUME)
        {
            WebView.ResumeTimers();
        }
        RunOnceResume();
    }
    

    OnDestroy 必须终止整个 WebView,如下所示:

    protected override void OnDestroy()
    {
        base.OnDestroy();
    
        WebView.RemoveAllViews();
        WebView.ClearHistory();
        WebView.ClearCache(true);
        WebView.LoadUrl("about:blank");
        WebView.OnPause();
        WebView.RemoveAllViews();
        WebView.PauseTimers();
        WebView.Destroy();
        WebView = null;
    }
    
  3. 判断是否是第一次的方法:

    private void RunOnceResume()
    {
        if (!RUN_ONCE_RESUME)
        {
            RUN_ONCE_RESUME = true;
        }
    }
    

我试过没有以前的方法,但它对我的情况不起作用。

通过所有这些更改,计时器将按预期工作。

关于javascript - 为什么在Android中旋转屏幕时setIntervals不能被杀死?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53091541/

相关文章:

javascript - 请求 POST 修改对象?

javascript - 如何根据文档执行与 vuejs 插槽通常执行的操作相反的操作?

java - 在android上运行java内的python脚本

java - 在 Netbeans 中的计时器中获取 JFrame 边界的问题

c# - 如何让定时器只触发一次

javascript - 测量元素的内容区域

javascript - 是否可以在 Adob​​e AIR 中使用 localStorage 和 window.openDatabase?

android - 对话框中的样式titleDivider

android - TangoService_connectOnFrameAvailable() 使用 Google Tango Leibniz Release 1.10 卡住或崩溃

javascript - 如何在 ASP.net 上正确使用启动和停止计时器?