我思故我强

JavaScript的执行顺序

通常状况下,javascript作为一种解释性的脚本,都是从上而下顺序执行的,但是JavaScript还允许在其语句中嵌套,也就是这个样子:

document.write("<script>alert(t);</scr"+"ipt>");

注意:</script>不能写作</script>,要用字符串连接的形式来表述,否则会有语法错误,估计是因为JavaScript遇到</script>就以为脚本结束了。

在这种情况下,正常的脚本和嵌入的脚本的执行顺序就需要研究一下了。



b.js:

alert("5");


a.js:

alert("4");
document.write("<script src=b.js></scr"+"ipt>");
alert("6");


test.html:

<script src=a.js></script>
<script>
alert("1");
document.write("<script src=b.js></scr"+"ipt>");
document.write("<script>alert("3")</scr"+"ipt>");
alert("2");
</script>



执行test.html,可以看到打印的顺序是:4,6,5,1,3,2,5

还可以做一些相关测试,得出的结论是:
1.同级的不同的代码块,代码块间的执行顺序为从上到下;
2.在代码中嵌入代码的情况下,先执行上层代码块,再执行子代码块;代码中嵌入代码是指一个文件引入另一个文件,而不是指所有的通过document.write形式打出的代码。
---------------------------------------------------------------------------------------------

一、在HTML中嵌入Javasript的方法

  1. 直接在Javascript代码放在标记对<script>和</script>之间
  2. 由<script />标记的src属性制定外部的js文件
  3. 放在事件处理程序中,比如:<p onclick="alert('我是由onclick事件执行的Javascript')">点击我</p>
  4. 作为URL的主体,这个URL使用特殊的Javascript:协议,比如:<a href="javascript:alert('我是由javascript:协议执行的javascript')">点击我</a>
  5. 利用javascript本身的document.write()方法写入新的javascript代码
  6. 利用Ajax异步获取javascript代码,然后执行

第3种和第4种方法写入的Javascript需要触发才能执行,所以除非特别设置,否则页面加载时不会执行。

二、Javascript在页面的执行顺序

  1. 页面上的Javascript代码是HTML文档的一部分,所以Javascript在页面装载时执行的顺序就是其引入标记<script />的出现顺序, <script />标记里面的或者通过src引入的外部JS,都是按照其语句出现的顺序执行,而且执行过程是文档装载的一部分。
  2. 每个脚本定义的全局变量和函数,都可以被后面执行的脚本所调用。
  3. 变量的调用,必须是前面已经声明,否则获取的变量值是undefined。
    <script type="text/javscrpt">//<![CDATA[
    alert(tmp);  //输出 undefined
    var tmp = 1;alert(tmp);  //输出 1
    //]]></script>
  4. 同一段脚本,函数定义可以出现在函数调用的后面,但是如果是分别在两段代码,且函数调用在第一段代码中,则会报函数未定义错误。
    <script type="text/javscrpt">//<![CDATA[
    aa(); //浏览器报错
    //]]></script>
    <script type="text/javscrpt">//<![CDATA[
    aa();   //输出 1
    function aa(){
       alert(1);}
    //]]></script>
  5. document.write()会把输出写入到脚本文档所在的位置,浏览器解析完documemt.write()所在文档内容后,继续解析document.write()输出的内容,然后在继续解析HTML文档。
    <script type="text/javascript">//<![CDATA[     
    document.write('<script type="text/javascript" src="test.js">
    <\/script>');     
    document.write('<script type="text/javascript">');     
    document.write('alert(2);')     
    document.write('alert("我是" + tmpStr);');     
    document.write('<\/script>');     
    //]]></script>   
    <script type="text/javascript">//<![CDATA[     
    alert(3);     
    //]]></script>

    test.js的内容是:

    var tmpStr = 1;      alert(tmpStr);
    • 在Firefox和Opera中的弹出值的顺序是:1、2、我是1、3
    • 在IE中弹出值的顺序是:2、1、3,同时浏览器报错:tmpStr未定义

    原因可能是IE在document.write时,并未等待加载SRC中的Javascript代码完毕后,才执行下一行,所以导致2先弹出,并且执行到document.write(’document.write("我是" + tmpStr)’)调用tmpStr时,tmpStr并未定义,从而报错。

    解决这个问题,可以利用HTML解析是解析完一个HTML标签,再执行下一个的原理,把代码拆分来实现:

    <script type="text/javascript">//<![CDATA[     
    document.write('<script type="text/javascript" src="test.js">
    <\/script>');     
    //]]></script>   
    <script type="text/javascript">//<![CDATA[     
    document.write('<script type="text/javascript">');     
    document.write('alert(2);')     
    document.write('alert("我是" + tmpStr);');     
    document.write('<\/script>');     
    //]]></script>   
    <script type="text/javascript">//<![CDATA[     
    alert(3);      //]]>
    </script>

    这样IE下和其他浏览器输出值的顺序都是一直的了:1、2、我是1、3。

三、如何改变Javascript在页面的执行顺序

  1. 利用onload
    <script type="text/javascript">//<![CDATA[
    window.onload = f;
    function f(){
        alert(1);
    }alert(2);
    //]]></script>

    输出值顺序是 2、1。

    需要注意的是,如果存在多个winodws.onload的话,只有最有一个生效,解决这个办法是:

    window.onload = function(){f();f1();f2();.....}

    利用2级DOM事件类型

    if(document.addEventListener){
    window.addEventListener('load',f,false);
    window.addEventListener('load',f1,false);...
    }else{
    window.attachEvent('onload',f);
    window.attachEvent('onload',f1);...
    }
  2. IE中可以利用deferdefer作用是把代码加载下来,并不立即执行,等文档装载完毕之后再执行,有点类似onload,但是没有onload那样的局限性,可以重复使用,但是只在IE中有效,所以上面的例子可以修改成为
    <script type="text/javascript">//<![CDATA[
    document.write('<script type="text/javascript" src="test.js"><\/script>');
    document.write('<script type="text/javascript" defer="defer">');
    document.write('alert(2);')
    document.write('alert("我是" + tmpStr);');
    document.write('<\/script>');
    //]]></script>
    <script type="text/javascript">//<![CDATA[
    alert(3);
    //]]></script>

    这样IE就不报错了,输出值的顺序变成:1、3、2、我是1

  3. 利用Ajax。
    因为xmlhttpRequest能判断外部文档加载的状态,所以能够改变代码的加载顺序。

posted on 2007-09-19 19:56 李云泽 阅读(2606) 评论(0)  编辑  收藏 所属分类: javascript


只有注册用户登录后才能发表评论。


网站导航: