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. unity3d 射弹基础案例

    小白本来想学cocos2dx的,然而c++难学就算了,cocos2dx对新手来说简直坑爹,于是乎转战unity3d学习js,在写出第一个游戏后兴致高多了哎. 回顾一下编辑的过程:1.建立一个cube作 ...

  2. URL Parsing

    [URL Parsing] urllib.parse.urlparse(urlstring, scheme='', allow_fragments=True) Parse a URL into six ...

  3. 分享一个 jquery serializeArray()序列化方法

    http://www.365mini.com/page/jquery-serializearray.htm http://www.365mini.com/diy.php?f=jquery-serial ...

  4. C# 使用Newtonsoft.Json.dll 格式化显示Json串

    private string ConvertJsonString(string str) { //格式化json字符串 JsonSerializer serializer = new JsonSeri ...

  5. css 层的嵌套

    html <div class="menu"> <ul> <li><a class="li1" title=" ...

  6. 清除SQLServer日志的两种方法

    日志文件满而造成SQL数据库无法写入文件时,可用两种方法:一种方法:清空日志.1.打开查询分析器,输入命令DUMP TRANSACTION 数据库名 WITH NO_LOG2.再打开企业管理器--右键 ...

  7. Xcode升级插件失效,与添加插件不小心点击Skip Bundle解决办法

    一.当发现升级xcode后,插件不能使用,解决办法如下: 1.查看Xcode的UUID 在终端执行 defaults read /Applications/Xcode.app/Contents/Inf ...

  8. cell线条前后缩短

    [cell setSeparatorInset:UIEdgeInsetsMake(0, 20, 0, 20)];

  9. HttpServletResponse

    Web服务器收到客户端的http请求,会针对每一次请求,分别创建一个用于代表请求的request对象.和代表响应的response对象.request和response对象即然代表请求和响应,那我们要 ...

  10. css控制页面打印(分页、屏蔽不需要打印的对象)

    样式: <style   media="print">     .Noprint   {   DISPLAY:   none;}     .PageNext   {   ...