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 相关文章推荐
总结AJAX相关JS代码片段和浏览器模型
Aug 15 Javascript
JavaScript表单常用验证集合
Jan 16 Javascript
js 利用className得到对象的实现代码
Nov 15 Javascript
JavaScript中DOM详解
Apr 13 Javascript
使用jquery+CSS3实现仿windows10开始菜单的下拉导航菜单特效
Sep 24 Javascript
jQuery插件EasyUI实现Layout框架页面中弹出窗体到最顶层效果(穿越iframe)
Aug 05 Javascript
js实现上传文件添加和删除文件选择框
Oct 24 Javascript
详解使用angular-cli发布i18n多国语言Angular应用
May 20 Javascript
微信小程序多音频播放进度条问题
Aug 28 Javascript
vue路由传参的基本实现方式小结【三种方式】
Feb 05 Javascript
原生js实现碰撞检测
Mar 12 Javascript
解决Vue的文本编辑器 vue-quill-editor 小图标样式排布错乱问题
Aug 03 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
php列出一个目录下的所有文件的代码
2012/10/09 PHP
Python中使用django form表单验证的方法
2017/01/16 PHP
PHP常见加密函数用法示例【crypt与md5】
2019/01/27 PHP
jquery miniui 教程 表格控件 合并单元格应用
2012/11/25 Javascript
JQueryEasyUI Layout布局框架的使用
2013/04/08 Javascript
简单漂亮的js弹窗可自由拖拽且兼容大部分浏览器
2013/10/22 Javascript
js动态设置div的值下例子
2013/10/29 Javascript
Jquery uploadify图片上传插件无法上传的解决方法
2013/12/16 Javascript
Javascript代码实现仿实例化类
2015/04/03 Javascript
js贪吃蛇网页版游戏特效代码分享(挑战十关)
2015/08/24 Javascript
jQuery form插件之ajaxForm()和ajaxSubmit()的可选参数项对象
2016/01/23 Javascript
Sea.JS知识总结
2016/05/05 Javascript
JavaScript轻松创建级联函数的方法示例
2017/02/10 Javascript
详解Node.js实现301、302重定向服务
2017/04/07 Javascript
使用jquery Ajax实现上传附件功能
2018/10/23 jQuery
微信小程序实现下拉菜单切换效果
2020/03/30 Javascript
Angular PWA使用的Demo示例
2019/01/31 Javascript
layer.confirm()右边按钮实现href的例子
2019/09/27 Javascript
小程序实现按下录音松开识别语音
2019/11/22 Javascript
js实现点击生成随机div
2020/01/16 Javascript
js实现坦克大战游戏
2020/02/24 Javascript
vue-cli4.x创建企业级项目的方法步骤
2020/06/18 Javascript
你不知道的 TypeScript 高级类型(小结)
2020/08/28 Javascript
python从入门到精通(DAY 3)
2015/12/20 Python
使用py2exe在Windows下将Python程序转为exe文件
2016/03/04 Python
python机器学习之神经网络(一)
2017/12/20 Python
python实现俄罗斯方块游戏
2020/03/25 Python
python中单下划线_的常见用法总结
2018/07/10 Python
Python批处理更改文件名os.rename的方法
2018/10/26 Python
python字典的setdefault的巧妙用法
2019/08/07 Python
Python绘制词云图之可视化神器pyecharts的方法
2021/02/23 Python
VisionPros美国站:加拿大在线隐形眼镜和眼镜零售商
2020/02/11 全球购物
META-INF文件夹中的MANIFEST.MF的作用
2016/06/21 面试题
2015年七七事变78周年纪念活动方案
2015/05/06 职场文书
总结一些Java常用的加密算法
2021/06/11 Java/Android
关于Python中*args和**kwargs的深入理解
2021/08/07 Python