移动Web与js定时器暂停或不准确计时的问题解决
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定时器暂停或不准确计时的问题解决的更多相关文章
- cocos2d-x JS 定时器暂停方法
this.scheduleOnce(function(){ this.addChild(Menugobtn);//要暂停执行的代码 }, 10);
- JS定时器不可靠的原因及解决方案
前言 在工作中应用定时器的场景非常多,但你会发现有时候定时器好像并没有按照我们的预期去执行,比如我们常遇到的setTimeout(()=>{},0)它有时候并不是按我们预期的立马就执行.想要知道 ...
- js定时器的使用(实例讲解)
在javascritp中,有两个关于定时器的专用函数,分别为: 1.倒计定时器:timename=setTimeout("function();",delaytime);2.循环定 ...
- js定时器 特定时间执行某段程序的例子
定时器想必大家并不陌生吧,在本文为大家详细介绍下js中是如何实现定时器的,具体原理及代码如下. 例子: $(function(){ var handler = function(){ //www.jb ...
- js定时器setInterval()与setTimeout()
js定时器setInterval()与setTimeout() 1.setTimeout(Expression,DelayTime),在DelayTime过后,将执行一次Expression,setT ...
- C#-WebForm JS定时器
JS定时器: 1.window.setTimeout(function(){},3000) 延迟3秒执行 2.window.setInterval(function(){},3000) 也叫重复器,每 ...
- 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. 不要 ...
- Vue清除所有JS定时器
Vue清除所有JS定时器 在webpack + vue 的项目中如何在页面跳转的时候清除所有的定时器 JS定时器会有一个返回值(数字),通过这个返回值我们可以找到这个定时器 在vue项目中可以使用路由 ...
- web前端js过滤敏感词
web前端js过滤敏感词 这里是用文本输入框还有文本域绑定了失去焦点事件,然后再遍历敏感词数组进行匹配和替换. var keywords=["阿扁","呵呵", ...
随机推荐
- JQuery 动画及一些小知识点
JQuery 动画 show(),hide()显示/隐藏slideDown(),slideUp() 拉开/合起fadeIn(),fadeOut()渐出/渐入自定义动画 animate({left:& ...
- const实现
[const实现]
- Oracle 取随机数(转)
1.从表中随机取记录 select * from (select * from staff order by dbms_random.random) where rownum < 4 ...
- struts1 html: textarea 不换行,变形
<html:textarea property="summary" style="word-wrap:break-word;word-break:break-all ...
- DrawerLayout学习,抽屉效果
第一节: 注意事项 *主视图一定要是DrawerLayout的第一子视图 *主视图宽度和高度匹配父视图,因为当你显示主视图时,要铺满整个屏幕,用户体验度较高 *必须显示指定的抽屉视图的android: ...
- iOS静态库.a文件制作和导入使用
iOS静态库.a文件制作: 1.新建Cocoa Touch Static Library工程 新建工程 - 选择iOS-FrameWork&Libary,选择 Cocoa Touch Stat ...
- SQLServer 脚本测试
最近在做大数据同步的工作.很少数据需要特殊清洗算法,每次测试,都测试全部数据,浪费时间,可以只测试那些特殊数据即可(切记).
- 【Mysql】phpMyAdmin安装与配置
phpMyAdmin概念 phpMyAdmin 是一个以PHP为基础,以Web-Base方式架构在网站主机上的MySQL的数据库管理工具,让管理者可用Web接口管理MySQL数据库.借由此Web接口可 ...
- adb server is out of date killing... 的解决办法
是adb server端口被占用了 你先执行adb nodaemon server ,查看adb server的端口是多少 1 2 C:\Users\xxxx>adb nodaemon serv ...
- GridView CheckBox 全选
GridView CheckBox 全选 <script type="text/javascript"> $(function () { $("#allChe ...