JavaScript执行顺序详细介绍

之前从JavaScript引擎的解析机制来探索JavaScript的工作原理,下面我们以更形象的示例来说明JavaScript代码在页面中的执行顺序。如果说,JavaScript引擎的工作机制比较深奥是因为它属于底层行为,那么JavaScript代码执行顺序就比较形象了,因为我们可以直观感觉到这种执行顺序,当然JavaScript代码的执行顺序是比较复杂的,所以在深入JavaScript语言之前也有必要对其进行剖析。
1.1  按HTML文档流顺序执行JavaScript代码
首先,读者应该清楚,HTML文档在浏览器中的解析过程是这样的:浏览器是按着文档流从上到下逐步解析页面结构和信息的。JavaScript代码作为嵌入的脚本应该也算做HTML文档的组成部分,所以JavaScript代码在装载时的执行顺序也是根据脚本标签<script>的出现顺序来确定的。例如,浏览下面文档页面,你会看到代码是从上到下逐步被解析的。

复制代码 代码如下:


<script>
alert("顶部脚本");
</script>
<html><head>
<script>
alert("头部脚本");
</script>
<title></title>
</head>
<body>
<script>
alert("页面脚本");
</script>
</body></html>
<script>
alert("底部脚本");
</script>

如果通过脚本标签<script>的src属性导入外部JavaScript文件脚本,那么它也将按照其语句出现的顺序来执行,而且执行过程是文档装载的一部分。不会因为是外部JavaScript文件而延期执行。例如,把上面文档中的头部和主体区域的脚本移到外部JavaScript文件中,然后通过src属性导入。继续预览页面文档,你会看到相同的执行顺序。

复制代码 代码如下:


<script>

alert("顶部脚本");

</script>

<html>

<head>

<script src="https://www.jb51.net/head.js"></script>

<title></title>

</head>

<body>

<script src="https://www.jb51.net/body.js"></script>

</body>

</html>

<script>

alert("底部脚本");

</script>

1.2  预编译与执行顺序的关系

在Javascript中,function才是Javascript的第一型。当我们写下一段函数时,其实不过是建立了一个function类型的实体。
就像我们可以写成这样的形式一样:

复制代码 代码如下:


functionHello()
{
alert("Hello");
}
Hello();
varHello = function()
{
alert("Hello");
}
Hello();


其实都是一样的。 但是当我们对其中的函数进行修改时,会发现很奇怪的问题。

复制代码 代码如下:


<scripttype="text/javascript">       
functionHello() {        
alert("Hello");  
}       
Hello();     
functionHello() {  
alert("Hello World");   
}       
Hello();   
</script>


我们会看到这样的结果:连续输出了两次Hello World。
而非我们想象中的Hello和Hello World。
这是因为Javascript并非完全的按顺序解释执行,而是在解释之前会对Javascript进行一次“预编译”,在预编译的过程中,会把定义式的函数优先执行,也会把所有var变量创建,默认值为undefined,以提高程序的执行效率。
也就是说上面的一段代码其实被JS引擎预编译为这样的形式:

复制代码 代码如下:


<scripttype="text/javascript">       
varHello = function() {          
alert("Hello");       
}      
Hello = function() {
           alert("Hello World"); 
      }     
Hello();    
Hello(); 
</script>


我们可以通过上面的代码很清晰地看到,其实函数也是数据,也是变量,我们也可以对“函数“进行赋值(重赋值)。
当然,我们为了防止这样的情况,也可以这样:

复制代码 代码如下:


<scripttype="text/javascript">     
functionHello() {       
alert("Hello");      
}       
Hello(); 
</script>  
<scripttype="text/javascript">  
functionHello() {         
alert("Hello World");    
}      
Hello(); 
</script>


这样,程序被分成了两段,JS引擎也就不会把他们放到一起了。   

当JavaScript引擎解析脚本时,它会在预编译期对所有声明的变量和函数进行处理。

做如下处理:

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/wdwjsp.html