高性能JavaScript:加载和执行

开发 前端
Javascript是一门很Web开发语言,和HTML,css是一样的,在本文中,我就自己看的一本书《高性能Javascript》所得的收获给大家分享一下。

       在步入正题之前为大家介绍一本很好的书《高性能JavaScript》大家不妨看一下,在本文中我为大家分享我看这本书的一些收获。

推荐的书

阻塞特性
       JS 有个很无语的阻塞特性,就是当浏览器在执行JS 代码时,不能同时做其他任何事情,无论其代码是内嵌的还是外部的。

脚本位置:
       浏览器在碰到一个引入外部JS 文件的<script>标签时会停下所有工作来下载并解析执行它,在这个过程中,页面渲染和用户交互完全被阻塞了,为了避免页面加载时的停顿甚至空白页的出现,JS 脚本应尽量放置在页面底部,这点很重要:

  1. <html>  
  2. <head>  
  3. <title>无标题文档</title>  
  4. <link rel="stylesheet" type="text/css" href="styles.css" />  
  5. </head>  
  6. <body>  
  7. <p>页面的内容。。。</p>  
  8. <!-- 推荐的位置,页面底部: -->  
  9. <script type="text/javascript" src="file1.js"></script>  
  10. <script type="text/javascript" src="file2.js"></script>  
  11. <script type="text/javascript" src="file3.js"></script>  
  12. </body>  
  13. </html> 

组织脚本:
       为了改善上面的阻塞情况,应尽可能的减少页面中<script>标签的出现次数,这同时也是考虑到HTTP 请求会带来额外的性能开销,也就是说应减少页面中外链脚本的数量。

       你可以手动合并你的多个JS 文件,也可采用类似Yahoo! combo handler 这样的实时在线服务来实现,例如下面的这个<script>标签实际上便载入了3个JS 文件:

  1. <html>    
  2. <head>    
  3. <title>无标题文档</title>    
  4. <link rel="stylesheet" type="text/css" href="styles.css" />    
  5. </head>    
  6. <body>      
  7. <p>页面的内容。。。</p>      
  8. <!-- 推荐的位置,页面底部: -->    
  9. <script type="text/javascript" src="http://yui.yahooapis.com/combo?file1.js&file2.js&file3.js"></script>    
  10. </body>    
  11. </html> 

无阻塞的脚本:
为了阻塞状况,这里提供了几个实现并行下载JS 脚本的方案。

1. 延迟的脚本

       HTML4 为<script>标签定义了一个defer 属性,它能使这段代码延迟执行,然而该属性只有IE4+ 和Firefox 3.5+ 支持。声明了defer 属性的<script>会在DOM加载完成,window.onload 事件触发前被解析执行:

  1. <script type="text/javascript" src="file1.js" defer></script> 

2. 动态脚本元素
       这是最通用的解决方案,通过DOM 动态地创建<script>元素并插入到文档中,文件在该元素被添加到页面时开始下载,这样 无论在何时启动下载,文件的下载和执行过程不会阻塞页面其他进程。

       不过要注意使用这种方式加载的代码会立刻执行,这样需清楚的了解各文件的作用以及合理的执行顺序,此时跟踪并确保脚本下载完成并准备就绪是很有必要的,非IE浏览器会在<script>元素接收完成时触发一个load 事件,而IE 下则会触发一个readystatechange 事件并通过readyState 属性加以判断便可。以下是兼容地动态加载一个JS 脚本的函数:了:

  1. function load_script(url, callback)   
  2. {      var script = document.createElement('script');        
  3.        script.type = 'text/javascript';  
  4.        if (script.readyState)   
  5.        {  //IE   
  6.          script.onreadystatechange = functio()
  7. {          
  8.  if (script.readyState == 'loaded' || script.readyState == 'complete'
  9. {
  10.                 script.onreadystatechange = null;
  11.                 callback();
  12.              }
  13.            }
  14.       } 
  15. else
  16. { //others
  17.         script.onload = function()
  18.              callback(); }
  19.       }
  20.       script.src = url;
  21.       document.getElementsByTagName('head')[0].appendChild(script);
  22.   } 

       你可以将这个函数保存至一个load_script.js 文件,然后用该函数来加载其他的脚本,当要加载多个脚本时,为了确保正确的加载顺序,可以将load_script() 的执行串联起来,最后如前面说到的放至页面的底部,这便是一个完美的解决方案了。

  1. <script  type="text/javascript"src="load_script.js"></script>    
  2. <script type="text/javascript">  
  3. load_script('file1.js'function()  
  4.  {     
  5.    load_script('file2.js'function()  
  6.      {        
  7.       load_script('file3.js', functio()  {                   
  8. //全部载入后的操作...           
  9.          } );  
  10.       } );  
  11.   } );  
  12. </script>  

3.XMLHttpRequest 脚本注入

       即通过AJAX 方式加载,不过这种方式无法实现跨域加载,不适用于大型网站。

推荐的无阻塞模式
       我们上面做的这些工作当然也已经被那些牛人们完成了,并写成了一些优秀的JS 类库以便我们使用,它们均能很好地解决JS 脚本的阻塞问题,实现并行下载,例如: YUI3、LazyLoad、LABjs 等。
 

【编辑推荐】

  1. JavaScript跨域总结与解决办法
  2. 使用Javascript开发移动应用程序
  3. 10个令人惊奇的HTML5和JavaScript效果
  4. JavaScript初学者应注意的七个细节
责任编辑:佚名 来源: 网络资源
相关推荐

2013-09-04 14:22:59

JavaScript性能优化

2011-04-19 11:06:03

JavaScriptweb

2014-10-10 14:00:52

JavascriptHTML

2015-07-23 15:17:37

JavaScript循环语句

2012-12-17 13:51:22

Web前端JavaScriptJS

2014-11-25 10:03:42

JavaScript

2009-06-24 15:00:39

Javascript代

2023-04-25 17:24:31

veImageXiOSSDK

2014-10-09 09:48:14

JavaScript

2013-06-06 10:53:39

Javascript装载

2009-01-05 10:00:11

JSP优化Servlet性能优化

2015-08-11 09:46:26

JavaScriptDOM编程重排

2012-03-12 09:33:04

JavaScript

2023-11-01 11:51:08

Linux性能优化

2019-08-26 18:20:05

JavascriptWeb前端

2017-12-07 13:40:00

JavaScript内存泄露内存管理

2019-03-01 11:03:22

Lustre高性能计算

2022-09-07 07:37:06

LIMITOFFSET分页

2017-11-28 17:14:16

华为云

2017-07-07 16:36:28

BIOIO模型 NIO
点赞
收藏

51CTO技术栈公众号