JavaScript计划任务后台运行的方法


Posted in Javascript onDecember 18, 2015

即使忘了 JavaScript 的一切知识,也不会忘记:它是阻塞的。

想象一下,你的浏览器里住着一个魔法小精灵,负责浏览器的正常运转。不论渲染 HTML,响应菜单命令,屏幕渲染,处理鼠标点击,或者执行 JavaScript 函数,所有事情都归一个小精灵处理。它哪忙得过来,一次只能处理一件事情。如果同时丢给它一堆任务,它会列一个长长的待办列表,按顺序完成它们。

人们常常希望初始化组件和事件处理的 JavaScript 可以尽快被执行。可是,有些不太重要的后台任务不会直接影响用户体验,比如:

记录统计数据

发送数据到社交网络(或添加‘分享'按钮)

预加载内容

预处理或预渲染 HTML

他们对时序要求不严格,但是为了让页面仍然响应,直到用户滚动页面或者与内容交互时才被执行。

选择之一是 Web Workers ,它可以在独立的线程同时执行代码。用于预加载和预处理再好不过,但是你没有权限直接访问或更新 DOM。你可以在自己的代码中避开这点,但是无法保证第三方脚本比如 Google Analytics 永远不需要这个。

另一个选择是setTimeout,比如setTimeout(doSomething, 1);。一旦其它的立即执行任务执行完毕,浏览器将执行doSomething()函数。实际上,它被放到了待办列表的底部。不幸的是,函数将被调用,而不顾处理需求。

#requestIdleCallback

requestIdleCallback 是新API,当浏览器稍作喘息的时候,用来执行不太重要的后台计划任务。 难免让人想起 requestAnimationFrame ,在下次重绘之前,执行函数更新动画。 想了解更多戳这里: 使用 requestAnimationFrame 做简单的动画 。

requestIdleCallback特性监测:

if ('requestIdleCallback' in window) {
 // requestIdleCallback supported
 requestIdleCallback(backgroundTask);
}
else {
 // no support - do something else
 setTimeout(backgroundTask1, 1);
 setTimeout(backgroundTask2, 1);
 setTimeout(backgroundTask3, 1);
}

也可以指定配置参数对象,比如 timeout,

requestIdleCallback(backgroundTask, { timeout: 3000; });

确保函数在3秒之内调用,不管浏览器是否空闲。

deadline对象传入以下参数时,requestIdleCallback仅执行一次回调:

didTimeout—— 如果可选的 timeout 触发,则设置为 true
timeRemaining()—— 函数返回执行任务剩余的毫秒数
timeRemaining()最多分配50ms用于任务的执行,超过这个限制,也不会停止任务,但是,最好重新调用requestIdleCallback安排进一步的处理。

我们来创建一个简单的例子,让几个任务按序执行。任务的函数引用储存在数组中:

//待执行的函数数组
var task = [
  background1,
  background2,
  background3
];
if ('requestIdleCallback' in window) {
 //支持 requestIdleCallback
 requestIdleCallback(backgroundTask);
}
else {
 //不支持 —— 立刻执行所有任务
 while (task.length) {
  setTimeout(task.shift(), 1);
 }
}
//requestIdleCallback 回调函数
function backgroundTask(deadline) {
 //如果存在,执行下一个任务
 while (deadline.timeRemaining() > 0 && task.length > 0) {
  task.shift()();
 }
 //需要的话,安排进一步任务
 if (task.length > 0) {
  requestIdleCallback(backgroundTask);
 }
}

#一次 requestIdleCallback 之间不应该做什么?

Paul Lewis 在 他的文章 中提到,一次 requestIdleCallback 执行的任务应该切成小块。它不适用于不可预知时间的情况(比如操作 DOM,使用 requestAnimationFrame 回调更好些)。resolving(或者 rejecting)Promises 时也要谨慎,即使没有更多的剩余时间,空闲回调完成之后,回调函数也将立即执行。

#requestIdleCallback 浏览器支持情况

requestIdleCallback是试验性特性,规范仍不稳定,碰到 API 变更时不足为奇。Chrome 47 已支持… 2015年结束前应该可用了。Opera 应该会紧跟其后。Microsoft 和 Mozilla 都在考虑 API 是否应该支持 Promises 。Apple 像往常一样不鸟。

Paul Lewis(上文提到的)写了一个简单的 requestIdleCallback shim ,它可以模拟浏览器的空闲监测行为,但不是一个 polyfill( shim 和 polyfill 的区别 )。

requestIdleCallback shim代码如下:

/*!
 * Copyright Google Inc. All rights reserved.
 *
 * Licensed under the Apache License, Version . (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-.
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
 * or implied. See the License for the specific language governing
 * permissions and limitations under the License.
 */
/*
 * @see https://developers.google.com/web/updates///using-requestidlecallback
 */
window.requestIdleCallback = window.requestIdleCallback ||
 function (cb) {
  var start = Date.now();
  return setTimeout(function () {
   cb({
    didTimeout: false,
    timeRemaining: function () {
     return Math.max(, - (Date.now() - start));
    }
   });
  }, );
 }
window.cancelIdleCallback = window.cancelIdleCallback ||
 function (id) {
  clearTimeout(id);
 }

ps: 如何运行计划任务

1、运行 GPEDIT.MSC

2、选择计算机配置

  --- Windows 设置
      --- 安全设置
        --- 本地策略
          --- 用户权利指派
双击右边的 从网络访问此计算机
把需要的用户名添加到列表。

3、    --- 安全设置

             ---安全选择
         开启允许服务器操作员计划任务

4、-----本地策略

    --- 作为批处理作业登录
     把需要的用户名添加到列表。

5、-----本地策略

   --- 允许计算机和用户被信任以便于委托
把需要的用户名添加到列表。
最好是administrator用户。

如果任务计划无法启动,提示代码:0X80041315

解决方法:这有两种可能,一是系统中的“Task Scheduler”服务没有启动,你可在运行中键入“services.msc”,查看“Task Scheduler”服务是否被设置成了“已禁用”,若是,只要双击它将启动类型改为“自动”,重新设定一个计划任务就可以执行了。

如果你当前账户设置了自动登录,而其登录密码又为空,也有可能导致任务计划不能按时执行,在XP专业版中,需要运行“gpedit.msc”来编辑组策略:展开“计算机配置→Windows设置→安全设置→本地计算机策略→安全选项”;双击右侧的“账户:使用空白密码的本地账户只允许进行控制台登录”项,在弹出对话框中选择“已禁用”。

Javascript 相关文章推荐
jquery中对表单的基本操作代码
Jul 29 Javascript
javascript获取作用在元素上面的样式属性代码
Sep 20 Javascript
js弹出div并显示遮罩层
Feb 12 Javascript
node.js中的fs.futimesSync方法使用说明
Dec 17 Javascript
javaScript中slice函数用法实例分析
Jun 08 Javascript
JavaScript检查子字符串是否在字符串中的方法
Feb 03 Javascript
Node.js操作mysql数据库增删改查
Mar 30 Javascript
关于Vue.js 2.0的Vuex 2.0 你需要更新的知识库
Nov 30 Javascript
使用vue实现grid-layout功能实例代码
Jan 05 Javascript
详解react内联样式使用webpack将px转rem
Sep 13 Javascript
傻瓜式vuex语法糖kiss-vuex整理
Dec 21 Javascript
js实现移动端图片滑块验证功能
Sep 29 Javascript
JavaScript jQuery 中定义数组与操作及jquery数组操作
Dec 18 #Javascript
原生js实现移动端瀑布流式代码示例
Dec 18 #Javascript
jQuery Validate表单验证入门学习
Dec 18 #Javascript
jQuery定义插件的方法
Dec 18 #Javascript
jQuery Validate插件实现表单强大的验证功能
Dec 18 #Javascript
jquery实现二级导航下拉菜单效果
Dec 18 #Javascript
学习jQuey中的return false
Dec 18 #Javascript
You might like
怎样辨别一杯好咖啡
2021/03/03 新手入门
php中的实现trim函数代码
2007/03/19 PHP
PHP 强制下载文件代码
2010/10/24 PHP
ThinkPHP让分页保持搜索状态的方法
2014/07/02 PHP
PHP中iconv函数知识汇总
2015/07/02 PHP
jQuery示例收集
2010/11/05 Javascript
从零开始学习jQuery (八) 插播:jQuery实施方案
2011/02/23 Javascript
jQuery 瀑布流 浮动布局(一)(延迟AJAX加载图片)
2012/05/23 Javascript
关于JS字符串函数String.replace()
2013/04/07 Javascript
js字符串截取函数substr substring slice使用对比
2013/11/27 Javascript
JS实现仿腾讯微博无刷新删除微博效果代码
2015/10/16 Javascript
jQuery定义插件的方法
2015/12/18 Javascript
jQuery实现图片加载完成后改变图片大小的方法
2016/03/29 Javascript
JavaScript排序算法动画演示效果的实现方法
2016/10/18 Javascript
Ionic3 UI组件之autocomplete详解
2017/06/08 Javascript
使用sessionStorage解决vuex在页面刷新后数据被清除的问题
2018/04/13 Javascript
微信小程序实现tab页面切换功能
2018/07/13 Javascript
在小程序开发中使用npm的方法
2018/10/17 Javascript
微信小程序封装多张图片上传api代码实例
2019/12/30 Javascript
[57:55]完美世界DOTA2联赛PWL S3 Magma vs Phoenix 第二场 12.12
2020/12/16 DOTA
在Python的struct模块中进行数据格式转换的方法
2015/06/17 Python
Python文件读取的3种方法及路径转义
2015/06/21 Python
python笔记:mysql、redis操作方法
2017/06/28 Python
详解Python 爬取13个旅游城市,告诉你五一大家最爱去哪玩?
2019/05/07 Python
pandas中DataFrame修改index、columns名的方法示例
2019/08/02 Python
图解python全局变量与局部变量相关知识
2019/11/02 Python
Python Web静态服务器非堵塞模式实现方法示例
2019/11/21 Python
Python的赋值、深拷贝与浅拷贝的区别详解
2020/02/12 Python
Python selenium页面加载慢超时的解决方案
2020/03/18 Python
7 For All Mankind官网:美国加州洛杉矶的高级牛仔服装品牌
2018/12/20 全球购物
诚信考试倡议书
2014/04/15 职场文书
项目建议书模板
2014/05/12 职场文书
飞越疯人院观后感
2015/06/09 职场文书
2017寒假社会实践心得体会范文
2016/01/14 职场文书
JavaScript实现淘宝商品图切换效果
2021/04/29 Javascript
浅谈Web Storage API的使用
2021/06/23 Javascript