欢迎来到[自学php网] ①群 AG亚游集团

AG亚游集团 > 网站前端 > javascript >

不会被 iOS 停掉的网页定时器方案

来源:未知 ?? 时间:2019-03-26 17:57?作者:小飞侠

[导读] 其实这个标题略微有点标题党:iOS 中,除了少数服务(如播放音乐),大部分 App 在用户按了 Home 键之后,过不了多久就会被完全冻结,这对 Safari 同样适用。本文不考虑这样情况,只...

其实这个标题略微有点标题党:iOS 中,除了少数服务(如播放音乐),大部分 App 在用户按了 Home 键之后,过不了多久就会被完全冻结,这对 Safari 同样适用。本文不考虑这样情况,只考虑 Safari 运行时,怎样让定时器持续工作。

我们知道:PC 上的 Firefox、Chrome 和 Safari 等浏览器,都会自动把未激活页面中的 JavaScript 定时器(setTimeout、setInterval)间隔最小值改为 1 秒以上。这是因为间隔很小的定时器一般用来做 UI 更新(例如用定时器实现的动画),让用户不可见的页面上的定时器跑慢一些,既节省资源又不会影响体验。对移动浏览器来说,内存、CPU、带宽等资源更加宝贵,移动设备上的浏览器往往会直接冻结所有未激活页面上的所有定时器。

我写了一个简单的 Demo 来说明这个问题:


<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8" />
    <meta content="width=320,minimum-scale=1.0,maximum-scale=1.0,user-scalable=no" name="viewport" />
</head>
<body>
    <div id="test" style="font-size:32px;"></div>
    <script>
        var count = 0,
            test = document.getElementById('test');
        setInterval(function() {
            count++;
            test.innerHTML = count;
            document.title = count;
        }, 1 * 1000);
    </script>
</body>
</html>

在装有 iOS7 的 iPad 下测试,可以很清楚看到定时器被冻结的现象:

setInterval on iPad

但是,这种策略有时也不那么完美。对 Web 邮箱、SNS 等网站来说,让用户及时知道有新消息很重要。很多网站会定时获取消息数,有新信息时在页面标题加上消息数,这样即使用户在浏览其它 Tab 时也可以看到提醒,这种弱提醒会给用户带来良好的产品体验。iOS 上,这种做法由于定时器被冻结而变得行不通了。

既然 iOS 的浏览器(我测试了 Safari 和 Chrome)都会冻结非激活页面的 JS 定时器,那么必须另辟蹊径了。今天刚好在这里看到,有个古老的页面刷新技术,无论页面是否可见都能刷新:<meta> 标签配合 refresh 属性。

简单介绍下这个 meta 头。假如页面 <head> 区有下面这行代码,页面会每隔 600s 刷新一次。

<meta http-equiv="refresh" content="600">

直接刷整个页面当然可以让页面更新,但也会把页面当前状态刷掉,还浪费流量,体验并不好。我们可以在页面引入 iframe,每次只刷新 iframe。这个 iframe 很小,流量是省了,但是每次都刷新还是会浪费 HTTP 请求,用强缓存又会使得更新很麻烦。有没有更好的办法呢?

这时候,轮到 Data URI 出场了。我们直接将含有 meta 刷新的页面 URL 编码,以 Data URI 的格式放到 iframe 的 src 中,就不会产生请求了。看下实际效果截图:

ios meta refresh

这个 Demo 的代码如下,iframe 跟主页面没有跨域,怎么传递消息都行。考虑本文讨论的是高级浏览器,我直接用的 postMessage:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8" />
    <meta content="width=320,minimum-scale=1.0,maximum-scale=1.0,user-scalable=no" name="viewport" />
    <title>后台页面更新测试 - meta refresh</title>
</head>
<body>
    <div id="test" style="font-size:32px;"></div>
    <script>
        var count = 0,
            test = document.getElementById('test');

        window.addEventListener('message', function(e) {
            if(e.data === 'refresh') {
                count++;
                test.innerHTML = count;
                document.title = count;
            }
        }, false);

        var duration = 1; /* 1s */

        var iframe = document.createElement('iframe');
        iframe.style.display = 'none';
        iframe.src = 'data:text/html,%3C%21DOCTYPE%20html%3E%0A%3Chtml%3E%0A%3Chead%3E%0A%09%3Cmeta%20charset%3D%22utf-8%22%20%2F%3E%0A%09%3Cmeta%20http-equiv%3D%22refresh%22%20content%3D%22'+ duration +'%22%20id%3D%22metarefresh%22%20%2F%3E%0A%09%3Ctitle%3Ex%3C%2Ftitle%3E%0A%3C%2Fhead%3E%0A%3Cbody%3E%0A%09%3Cscript%3Etop.postMessage%28%27refresh%27%2C%20%27%2A%27%29%3B%3C%2Fscript%3E%0A%3C%2Fbody%3E%0A%3C%2Fhtml%3E';
        document.body.insertBefore(iframe, document.body.childNodes[0]);
    </script>
</body>
</html>

在装有 iOS7 的 iPhone 上测试,也没问题。只是 iPhone 的 Safari 在预览多标签的时候,显示的是页面的截图,所以下图中虽然整个页面一直在更新,却只能看到 Title 的变化。

iphone meta refresh

实际上,Chrome PC 版支持 <meta http-equiv="refresh" content="0.1"> 这样的小于 1 的刷新间隔,所以这种方案也可以让 Chrome PC 版不可见页面的定时器间隔小于 1s。只是浏览器的刷新按钮会闪个不停,估计没人忍受得了。

最后,在 iOS 中,使用 <meta> 方案模拟的定时器,如果需要仅在 Tab 未激活时才刷新,也很简单。代码如下:

<meta http-equiv="refresh" content="10" id="refresh">
<script>
    var meta = document.getElementById("refresh");

    setInterval(function() {
        meta.content = meta.content;
    }, parseInt(meta.content / 2) * 1000);
</script>

首先获取 meta 的刷新间隔,再设置一个间隔稍短的定时器,不断重置 meta 的刷新间隔。这样只要页面处于激活状态,meta 就没机会刷新。而页面非激活时,由于 setInterval 被冻结,页面又可以被 meta 刷新了。


最新文章

点击排行

自学PHP网专注网站建设学习,PHP程序学习,平面设计学习,以及操作系统学习

京ICP备14009008号@版权所有AG亚游集团

网站声明:本站所有视频,教程都由网友上传,站长收集和分享给大家学习使用,如由牵扯版权问题请联系站长邮箱904561283@qq.com

联想要誓死打赢的保卫战 背后是怎样的真相与博弈 百度对5.5万个医院名称进行品牌保护 日媒羡慕中国或成AI第一大国:日本或\"落后一代人\" 日媒称安倍与斯里兰卡总统谈海洋安保 有意针对中国 海南房地产调控再加码:岛内实行全域限购 台湾学霸蜂拥来大陆求学 他们都是怎么想的? 一图揭示全球最负盛名的50强公司发展趋势 第一上海:港股伸延上攻势头 走了一波三连涨行情 浙江队确认向韦德报价:希望科比事件不要重演 液化天然气亚洲价格下跌 因中国需求减少 关小刀任九:瓦朗谢讷主场拿分 埃瓦尔客战难胜 [新浪彩票]足彩18047期任九:罗马主胜可博
印尼塞兰海发生5.9级地震 震源深度20千米 社科院副院长李培林:要防止对社会阶层流动的阻碍 香港冬季流感高峰期正式结束 共导致382人死亡 佩雷拉:有实力不在乎年龄 未来国足有陈彬彬位置 俄汉学家齐赫文斯基去世 曾亲历斯大林毛泽东会谈 金时沅重回福地锯齿草 期待5月卫冕球员锦标赛 日本正式引进外国介护技能实习生 2名中国人入选 穆帅豪砸6500万英超铁闸遭拒 改撬巴萨失意巨兽 迪拜公司否认侵犯阿里巴巴商标权:名字取自中东传说 涉嫌贩卖毒品嫌犯未被警方提请逮捕 检方予以追捕 本以为是波狗粮,谁想到结局让单身狗无比舒适 上港足协杯生死战将变阵 浩克停赛埃神迎两月首战
5岁女童因1个动作双下肢瘫痪 学这些孩子没到年龄 华为宣布扩大印度手机产能 目标10%市场占有率 Twitter第一季度净利润1.23亿美元 同比增长1… 77球!利物浦造欧洲最疯三叉戟 皇马拜仁也颤抖 风投支持金融科技独角兽全球排名:陆金所第一 浙媒:广厦\"打怪\"攒经验升级 装备差输最终boss 意见箱离地约2米 媒体:要配个小凳子来提意见? 抢十赛张帅胜小威进决赛憾负 无缘25万美元奖金 环球时报社评:即使中美贸易归零 中国也不会后退 自己在家怎么挣钱 白手起家创业做什么 广东水库抓到一只女鬼 怎么白手起家 AG亚游集团