PC 上的 Firefox、Chrome 和 Safari 等浏览器,都会自动把未激活页面中的 JavaScript 定时器(setTimeout、setInterval)间隔最小值改为 1 秒以上;而移动设备上的浏览器往往会直接冻结未激活页面上的所有定时器」。今天继续聊一聊 JavaScript 定时器与移动 Web 这个话题。

计时器

最简单的计时器只需要一个时间变量和固定间隔运行的函数就可以了,定期把上一次时间(默认为系统初始时间)加上运行间隔就是当前时间了。在 PC 上,这样实现的计时器只要运行间隔没有小于 1s,多半没什么大问题。但移动端不一样,只要页面不在前台显示,计时器就彻底不走了。

这个问题很容易解决,只要把定时器函数里的时间累加逻辑,换成每次都从系统时间获取就可以了。那如果系统时间不准怎么办?实际上大部分智能手机,默认都会开启网络校时,而且很多人有拿手机当手表的习惯,所以移动 Web 里获取到的系统时间,相比 PC 端,要准确得多。

更严谨的做法是服务端输出页面时带上服务器时间,JS 计时器先算出这个时间与系统时间之差 Δt,之后每次都用当前系统时间 + Δt 来还原服务器时间。逻辑很简单不多介绍了,说两个问题:

1)如果定时器运行时,用户改了系统时间怎么办?这个问题有很多解决方案,但是在移动设备上却有个偷懒的办法,由于移动设备上修改系统时间几乎一定会让页面进入后台,所以我们只需要在页面「从后台切回」时刷新下页面就好了。

2)移动设备在 2G 等网络环境下,接收响应需要更长时间,很有可能 JS 从页面上拿到服务器时间,已经是几十秒之后的事情了,丧失了准确性。如果一定要解决这个问题,使用 HTML5 新增的 Navigation Timing API 来修正是一种思路。美中不足的是 Android 4.0+ 才支持,Safari 系列则从未支持过这个 API。

从后台切回

要判断移动端页面是否从后台切回有很多方案,但利用后台页面定时器被冻结这个特性,有个简便且通用的做法:

var lastTime = +new Date;
setInterval(function() {
if(Math.abs(+new Date - lastTime) > 3000) {
alert('从后台切回!');
}
lastTime = +new Date;
}, 1000);

原理很简单,每隔一段时间(如 1s)更新页面上某个变量,如果下次这个变量与当前系统时间之差大于某个阈值(如 3s),说明页面定时器肯定被冻结过,也就是说页面是从后台切回来。代码中加上 Math.abs 是因为用户可能把系统时间往回改(当然极端情况下还是会检测不出来,忽略就好了)。

另一种定时器

「不会被 iOS 停掉的网页定时器」,原理是利用不会被 iOS 冻结的<meta> 标签 refresh 功能模拟 JS 定时器,有兴趣的同学可以点这里了解下。

本文链接:https://www.imququ.com/post/mobile_web_and_js_timer.html

此文转载自:JerryQu 的小站

移动Web与js定时器暂停或不准确计时的问题解决的更多相关文章

  1. cocos2d-x JS 定时器暂停方法

    this.scheduleOnce(function(){ this.addChild(Menugobtn);//要暂停执行的代码 }, 10);

  2. JS定时器不可靠的原因及解决方案

    前言 在工作中应用定时器的场景非常多,但你会发现有时候定时器好像并没有按照我们的预期去执行,比如我们常遇到的setTimeout(()=>{},0)它有时候并不是按我们预期的立马就执行.想要知道 ...

  3. js定时器的使用(实例讲解)

    在javascritp中,有两个关于定时器的专用函数,分别为: 1.倒计定时器:timename=setTimeout("function();",delaytime);2.循环定 ...

  4. js定时器 特定时间执行某段程序的例子

    定时器想必大家并不陌生吧,在本文为大家详细介绍下js中是如何实现定时器的,具体原理及代码如下. 例子: $(function(){ var handler = function(){ //www.jb ...

  5. js定时器setInterval()与setTimeout()

    js定时器setInterval()与setTimeout() 1.setTimeout(Expression,DelayTime),在DelayTime过后,将执行一次Expression,setT ...

  6. C#-WebForm JS定时器

    JS定时器: 1.window.setTimeout(function(){},3000) 延迟3秒执行 2.window.setInterval(function(){},3000) 也叫重复器,每 ...

  7. atitit.GUI图片非规则按钮跟动态图片切换的实现模式总结java .net c# c++ web html js

    atitit.GUI图片非规则按钮跟动态图片切换的实现模式总结java .net c# c++ web html js 1. 图片按钮的效果总结 1 1.1. 按钮图片自动缩放的. 1 1.2. 不要 ...

  8. Vue清除所有JS定时器

    Vue清除所有JS定时器 在webpack + vue 的项目中如何在页面跳转的时候清除所有的定时器 JS定时器会有一个返回值(数字),通过这个返回值我们可以找到这个定时器 在vue项目中可以使用路由 ...

  9. web前端js过滤敏感词

    web前端js过滤敏感词 这里是用文本输入框还有文本域绑定了失去焦点事件,然后再遍历敏感词数组进行匹配和替换. var keywords=["阿扁","呵呵", ...

随机推荐

  1. CSS-页面布局

    介绍 几个实现多栏布局的方法.主要介绍使用内部div来创建浮动的栏. 多栏布局有三种基本的实现方案:固定宽度.流动.弹性. 固定宽度布局的大小是随用户调整浏览器窗口大小而变化,一般是900至1100像 ...

  2. nginx环境下配置nagios-关于commands.cfg

    -w $ARG1$ -c $ARG2$ -M -b% -c % -f% -c % -f% -c % -f #  define command{         command_name    chec ...

  3. Android文件系统的结构

    Android 4.2.2 版本的文件系统 内核版本为 3.0.31 版本号为JDQ39 factory//估计是存放网络通信协议的登录密钥的|-- bluetooth|-- hdcp.keys|-- ...

  4. Lua metatable & metamethod

    [Lua metatable & metamethod] Lua中的每个值都有一套预定义的操作集合,比如数字是可以相加的,字符串是可以连接的,但是对于两个table类型,则不能直接进行“+”操 ...

  5. nested exception is com.mysql.jdbc.PacketTooBigException: Packet for query is too large (1044 > 1024

    HTTP Status 500 - type Exception report message description The server encountered an internal error ...

  6. jQuery数组处理汇总

    jQuery数组处理汇总   有段时间没写什么了, 打算把jquery中的比较常用的数组处理方法汇总一下 $.each(array, [callback])遍历,很常用 1 2 3 4 5 6 7 8 ...

  7. 在Windows上运行Tomcat

    一.下载jdk 百度搜索jdk 64位 下载,直接就有链接.或者官网下载 二.下载tomcat http://tomcat.apache.org/ 三.安装jdk并配置环境 安装jdk后进入控制面板- ...

  8. 笔记本Linux推荐

    1.CUB LINUX Cub Linux 是一个最好的选择.他的前身来自著名的 Chromium OS , Cub Linux 能够运行在各种各样的笔记本上面.即便是早年的老机,亦或是现在的新机.从 ...

  9. 技术英文单词贴--V

    V validate 验证,确认,使生效 verify 核实,查实,验证 version 版本,译文 via 通过,凭借,经过 prep

  10. CSS3动画与2D、3D转换

    一.过度动画:transition 五个属性: transition-property css 样式属性名称 transition-duration 动画持续时间(需要单位s) transition- ...