JavaScript 在各个浏览器中执行的耐性


Posted in Javascript onApril 06, 2009

IE:执行超过500W条JScript引擎语句出现提示。
Firefox:执行超过10秒出现提示。
Safari:执行超过5秒出现提示。
Opera:无论执行多久都不会出现提示,最有耐性。
Chrome:执行超过约8秒(估计值)出现提示。
注:当弹出类似alert的模式对话框的时候,是不计时。

Web开发的时候,经常会遇到的一种情况就是浏览器提示脚本运行时间过长,停止还是继续,无论你选择什么,相信你都会想尽一切办法让这个对话框远离你的用户们。可你是否知道,这些不同的浏览器究竟是如何判断,哪些脚本处于“失控”状态么?本文作者,就从Internet Explorer、Firefox、Safari、Chrome和Opera五种浏览器,分析了这个情况出现的原因。

【原文标题】What determines that a script is long-running?
【原文作者】Nicholas C. Zakas

以下是对原文的翻译:

Web开发者经常遇到并必须及时处理的问题就是“提示脚本运行时间过长的提示框”(或者称为“失控脚本提示”),这些令人讨厌的对话框会在你的脚本 执行时间过长的时候出现。对于Web开发者的基本准则就是,无论什么时候,都不要让用户看到这些对话框,因为这会给人一种代码缺乏结构化的印象,更简单的 说,你的代码负担太重了。

用Brendan Eich(JavaScript的发明人)的话来讲,如果JavaScript运行的时间需要用秒来计算,一定是什么地方搞错了。我个人可以忍受的上限可 能更小一些,不论什么脚本,在任何时间、任何浏览器上执行,都不应该超过100毫秒。如果实际执行的时间长于这个底限,一定要将进程分解成若干更小的代码 段。

另外,其实很少有人真正意识到究竟是什么原因导致脚本在不同的浏览器中运行时间过长,连我自己都没有深究过。所以我决定坐下来好好研究一下,我们究 竟会在什么情况才会看到那个讨厌的对话框。判断脚本是否失控,无外乎就两种方法。一种是根据执行了多少条语句,一种是判断脚本执行花费的时间。各个浏览器 判断脚本失控的具体方法会有略微的不同。

Internet Explorer

Internet Explorer判断一个脚本是否失控,主要通过JScript引擎执行语句的总数来判断。默认情况下,这个上限是500万条语句,这个值是可以通过注册表修改的。当你的脚本执行的语句数量超过这个限制,你就会看到下面的窗口。

JavaScript 在各个浏览器中执行的耐性

这个对话框提示:“这个页面上有一段脚本导致Internet Explorer运行缓慢,如果你继续运行,你的计算机可能会变为无响应状态”。要不是追求技术上的准确性,这样说确实有点过了。对话框有两个选项,要么 停止脚本执行,要么允许脚本继续运行。当这个对话框显示的时候,脚本已经被完全停止了。如果你选择继续运行脚本,就会重新计算当前执行的语句数,也就是 说,如果这个数值再次达到上限时,你会再次看到这个对话框。

Firefox

Firefox是根据脚本引擎持续执行代码的时间来判断一段脚本是否失控。默认的上限是10秒,可以通过about:config页面来修改这个值。这里需要注意的是,当弹出类似alert的模式对话框的时候,是不计时的。当浏览器执行脚本的时间达到这个上限,Firefox就会显示类似下面的对话框:

JavaScript 在各个浏览器中执行的耐性

Firefox的对话框提示:“这个页面的一段脚本目前运行忙,或者这段脚本已经停止响应。你可以停止执行这段脚本,并在调试器中打开这段脚本,或 者保持这段脚本继续运行”。更清楚的描述了遇到的问题,并且没有IE说的那么恐怖。在这个对话框上可以执行三种操作:停止脚本执行、调试脚本或者让脚本继 续运行。和Internet Explorer一样,当运行脚本继续运行以后,对持续运行脚本时间的统计就会重置。调试脚本按钮,只有在你安装了Firebug,并在该页面激活了调试 的时候才会出现。执行调试脚本操作后,可以显示执行时间过长的代码段的具体位置。

Safari

Safari同样根据脚本引擎持续执行脚本的时间来判断,当我对Webkit的源代码进行反复研究后,发现默认的超时时间是5秒,一旦达到这个上限,就会给出下面的对话框提示:

JavaScript 在各个浏览器中执行的耐性

对话框提示:“在页面url上的脚本让Safari失去响应,你是要继续运行脚本还是终止脚本”。同样的,对于用户来说,也不是什么可怕的提示。在Safari中,可以关闭失控脚本的检测功能

Chrome

Chrome在跟踪技术上有点狡猾,失控脚本检测功能似乎和tab的事故控制(crash control)关联到一起。我仔细看了源代码,却没有找到具体的限制,但基本确定的是,这个限制是以时间为基础的,估计在10秒左右(要么是5秒,要么 是10秒,总要和Safari或者Firefox看齐么)。我正在联系Chrome项目组中的朋友,看看能不能得到确定的信息。尽管如此,如果网页中存在 失控的脚本,用户还是会看到下面的对话框:

JavaScript 在各个浏览器中执行的耐性

毫无疑问,Chrome的提示比起其他浏览器来说,显得都更加严重。点击“Wait”按钮,脚本会继续运行,直到达到下一个上限为止,也可以点击“Kill pages”,直接关闭该页面在内存中的所有信息,并用一个空白页取而代之。

Opera

Opera的情况比较有趣:他貌似没有针对失控脚本的相应限制。我运行了几个很长的测试,甚至花了几分钟,而在这个过程中,浏览器一直可以正常响应,这很出我的意料之外。我不是很确定,对于现在的情况来说,这个方法是好是坏,但至少它生效了,不是么?

一些建议

无论你的用户使用什么浏览器,都不应该在任何时候看到类似的提示。在你的网站或者Web应用程序作为产品发布之前,做一些常规的性能测试是非常有必要的。在这方面有很多工具可以加以利用,比如Firebug's profiler(只支持Firefox)、YUI Profiler (支持全部浏览器)或者Internet Explorer 8's Profiler。 你应该毫不犹豫地将那些执行时间超过100毫秒的脚本找出来,哪怕这些脚本只是在某些浏览器上运行不畅,这些脚本包含了一些需要执行很长时间的代码段,而 这些代码应该通过性能检测工具进行重新评估。确保你不是使用Chrome作为测试的底线,因为Chrome在执行JavaScript的速度上比其他浏览 器要高出一个数量级(和Firefox 3.1还有最新的WebKit Nightly相当)。最好使用Internet Explorer作为测试的底线,然后再测试其他浏览器,因为无论什么时候,IE的JavaScript引擎都是最慢的,当在IE上修复问题以后,十有八 九在其他浏览器上也可以正常运行了。

Javascript 相关文章推荐
用JS写的一个TableView控件代码
Jan 23 Javascript
jQuery ctrl+Enter shift+Enter实现代码
Feb 07 Javascript
jsPDF生成pdf后在网页展示实例
Jan 16 Javascript
javascript实现表格增删改操作实例详解
May 15 Javascript
js事件处理程序跨浏览器解决方案
Mar 27 Javascript
jquery pagination分页插件使用详解(后台struts2)
Jan 22 Javascript
Node.js websocket使用socket.io库实现实时聊天室
Feb 20 Javascript
基于openlayers4实现点的扩散效果
Aug 17 Javascript
搭建element-ui的Vue前端工程操作实例
Feb 23 Javascript
Angular2进阶之如何避免Dom误区
Apr 02 Javascript
vue输入节流,避免实时请求接口的实例代码
Oct 30 Javascript
解决elementUI 切换tab后 el_table 固定列下方多了一条线问题
Jul 19 Javascript
javascript 获取图片颜色
Apr 05 #Javascript
Mozilla 表达式 __noSuchMethod__
Apr 05 #Javascript
关于javascript document.createDocumentFragment()
Apr 04 #Javascript
HTML 自动伸缩的表格Table js实现
Apr 01 #Javascript
Javascript 原型和继承(Prototypes and Inheritance)
Apr 01 #Javascript
说说掌握JavaScript语言的思想前提想学习js的朋友可以看看
Apr 01 #Javascript
setTimeout 不断吐食CPU的问题分析
Apr 01 #Javascript
You might like
实用PHP会员权限控制实现原理分析
2011/05/29 PHP
PHP的异常处理类Exception的使用及说明
2012/06/13 PHP
Linux中用PHP判断程序运行状态的2个方法
2014/05/04 PHP
Laravel 5框架学习之日期,Mutator 和 Scope
2015/04/08 PHP
onpropertypchange
2006/07/01 Javascript
简明json介绍
2008/09/28 Javascript
javascript检查日期格式的函数[比较全]
2008/10/17 Javascript
this和执行上下文实现代码
2010/07/01 Javascript
JS操作select下拉框动态变动(创建/删除/获取)
2013/06/02 Javascript
JS 获取浏览器和屏幕宽高等信息代码
2014/03/31 Javascript
jQuery中:selected选择器用法实例
2015/01/04 Javascript
jQuery中end()方法用法实例
2015/01/08 Javascript
Javascript核心读书有感之类型、值和变量
2015/02/11 Javascript
JS中捕获console.log()输出的方法
2015/04/16 Javascript
jQuery插件Validation快速完成表单验证的方式
2016/07/28 Javascript
利用jquery实现瀑布流3种案例
2016/09/18 Javascript
jQuery的deferred对象使用详解
2016/09/25 Javascript
Node.js如何实现注册邮箱激活功能 (常见)
2017/07/23 Javascript
ES6扩展运算符用法实例分析
2017/10/31 Javascript
微信小程序中悬浮窗功能的实现代码
2019/08/02 Javascript
vue弹出框组件封装实例代码
2019/10/31 Javascript
[43:32]2014 DOTA2华西杯精英邀请赛 5 25 LGD VS NewBee第一场
2014/05/26 DOTA
Pthon批量处理将pdb文件生成dssp文件
2015/06/21 Python
Python的collections模块中namedtuple结构使用示例
2016/07/07 Python
使用11行Python代码盗取了室友的U盘内容
2018/10/23 Python
解决pyqt5中QToolButton无法使用的问题
2019/06/21 Python
python tkinter库实现气泡屏保和锁屏
2019/07/29 Python
python中使用while循环的实例
2019/08/05 Python
Python使用matplotlib 画矩形的三种方式分析
2019/10/31 Python
香港个人化生活购物网站:Ballyhoo Limited
2016/09/10 全球购物
blueseventy官网:铁人三项和比赛泳衣
2021/02/06 全球购物
2014年征兵标语
2014/06/20 职场文书
党员民主评议个人总结
2014/10/20 职场文书
钱学森观后感
2015/06/04 职场文书
2016孝老爱亲模范事迹材料
2016/02/26 职场文书
Python中glob库实现文件名的匹配
2021/06/18 Python