探讨JavaScript语句的执行过程


Posted in Javascript onJanuary 28, 2016

废话不多说,直奔主题了。javascript的运行原理总结如下:

1、按照html文档流顺序执行javascript代码

浏览器是按照文档流从上到下逐步解析页面结构和信息的,javascript代码作为嵌入的脚本作为html文档的组成部分,所以javascript代码在加载时的执行顺序也是根据脚本标签<script>的出现顺序来确定的。

如果通过脚本标签<script>的src属性来引入外部.js文件,那么它也将按照其语句出现的顺序来执行,而且执行过程是文档加载的一部分。不会因为是外部js文件而延期执行。

2、预编译和执行顺序的关系

首先看如下这段代码:

<script type="text/javascript">
function hello() {
alert("hello");
}
hello();
function hello() {
alert("hello world");
}
hello();
</script>

上面这段js代码的输出结果是hello world 、hello world,而不是先输出hello,再输出hello world。这是因为javascript并非完全按照顺序来解释执行,而是在解释之前会对javascript进行一次预编译,在预编译的过程中,会把定义式的函数优先执行,也会把所有var变量创建,默认值为undefined,以提高程序的执行效率。也就是说上面的这段代码其实被JS引擎预编译成下面这样:

<script type="text/javascript">
var hello = function() {
alert("hello");
};
hello = function() {
alert("hello world");
};
hello();
hello();
</script>

通过上面的代码可以清晰的看到,函数其实也是变量,可以对函数进行赋值。为了防止前面那种情况的出现,可以如下定义成两个js文件:

<script type="text/javascript">
hello();
function hello() {
alert("hello");
}
// hello();
</script>
<script type="text/javascript">
function hello() {
alert("hello world");
}
hello();
</script>

上面第一个文件,我把hello()放在了function的前面,也是可以输出正确结果的。

<script type="text/javascript">
hello();
var hello = function() {
alert("hello");
};
// hello();
</script>

如果用上面的这种方法对function函数进行定义,那么就会报错,报错信息如下图1所示:

 

这里报错hello is not a funtion,这是由于在预编译的时候,对于用var声明的变量,虽然最先就处理了,但是变量值是undefined。然后运行hello()的时候,由于前面的hello是undefined,类型没有确定,所以这里是hello is not a function。虽然,程序中有定义这个函数,但是定义的位置放在了调用的后面,那么调用的时候,程序并没有运行到这里,所以没用。

再来看下面的这一段代码:

<script type="text/javascript">
hello();
function hello() {
alert("hello");
}
// hello();
</script>

上面的这段代码虽然调用也是在函数定义的前面,但是这里是以function关键字来定义的,用function来定义的时候,跟var不一样,function定义的时候就已经把函数的值赋了过去,所以这里可以运行。

总结:

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

(1)在执行前会进行类似“预编译”的操作:首先会创建一个当前执行环境下的活动对象,并将那些用var声明的变量设置为活动对象的属性,但是此时这些变量的赋值都是undefined,并将那些以function定义的函数也添加为活动对象的属性,而且它们的值正是函数的定义。

(2)在解释执行阶段,遇到变量需要解析时,会首先从当前执行环境的活动对象中查找,如果没有找到而且该执行环境的拥有者有prototype属性时则会从prototype链中查找,否则将会按照作用域链查找。遇到var a = ...这样的语句时会给相应的变量进行赋值(注意:变量的赋值是在解释执行阶段完成的,如果在这之前使用变量,它的值会是undefined)。

(3)综上,一句话总结就是:变量的声明在预编译期,变量的初始化在运行期。

<script type="text/javascript">
alert(a); // 在预编译期间a变量已经加载,但是用var定义,所以赋值为undefined先,故这里输出undefined。
var a = 1; // 这里给前面的没有赋值的a进行赋值为1
alert(a); // 这里输出的a已经是前面赋值过的,所以输出1。
</script>

对于上面的这段代码,输出结果是:先输出undefined,后输出1,分析见代码备注。

虽然变量和函数声明可以在文档任意位置,但是良好的习惯应该是在所有JavaScript代码之前声明全局变量和函数,并对变量进行初始化赋值。在函数内部也是先声明变量,然后再引用。

3、按块执行javascript代码

所谓代码块就是使用<script>标签分隔的代码段。JavaScript解释器在执行脚本时,是按块来执行的。通俗地说,就是浏览器在解析HTML文档流时,如果遇到一个<script>标签,则JavaScript解释器会等到这个代码块都加载完后,先对代码块进行预编译,然后再执行。执行完毕后,浏览器会继续解析下面的HTML文档流,同时JavaScript解释器也准备好处理下一个代码块。由于JavaScript是按块执行的,所以如果在一个JavaScript块中调用后面块中声明的变量或函数就会提示语法错误。

<script>
alert(a);
</script>
<script>
var a = 1;
</script>

上面的这段代码,由于是两个代码块,先执行完第一个代码块,再执行第二个代码块。执行第一个代码块的时候,变量a没有声明,所以报错,报错信息是:a is not defined。

<script>
var a = 1;
</script>
<script>
alert(a);
</script>

虽然说,JavaScript是按块执行的,但是不同块都属于同一个全局作用域,也就是说,块之间的变量和函数是可以共享的。所以,上面的这两个代码块运行的时候,虽然是两个代码块,但是第一段运行以后,a变量就存在了全局作用域中,此时运行到第二个代码块,输出的a变量就可以调用全局作用域中的a,所以没有问题。

4、借助事件机制改变javascript执行顺序

由于JavaScript是按块处理代码,同时又遵循HTML文档流的解析顺序,所以在上面示例中会看到这样的语法错误。但是当文档流加载完毕,如果再次访问就不会出现这样的错误。为了安全起见,我们一般在页面初始化完毕之后才允许JavaScript代码执行,这样可以避免网速对JavaScript执行的影响,同时也避开了HTML文档流对于JavaScript执行的限制。

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>javascript</title>
<script>
window.onload = function() {
alert(a);
};
</script>
<script>
var a = 1;
alert("bb");
</script>
</head>
<body>
</body>
<script>
alert("cc");
</script>
</html>

windows.onload = function()表示先在触发事件上加一个函数,并不立即执行,而是在整个页面都加载完成以后再开始执行该事件,及function。所以,在windows.onload执行之前,就已经把一些变量加载到了全局区中,所以没有问题。上面的输出结果是:先输出bb,再输出cc,最后输出a。

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>javascript</title>
<script>
window.onload = function() {
alert(a);
};
// 上面的onload不会执行,只会执行下面的onload
window.onload = function() {
alert("onload2");
};
</script>
<script>
var a = 1;
alert("bb");
</script>
</head>
<body>
</body>
<script>
alert("cc");
</script>
</html>

如果在一个页面中存在多个windows.onload事件处理函数,则只有最后一个才是有效的(如上面的代码所示),为了解决这个问题,可以把所有脚本或调用函数都放在同一个onload事件处理函数中,如下面的代码所示:

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>javascript</title>
<script>
window.onload = function() {
// 放到一起
alert(a);
alert("onload2");
};
</script>
<script>
var a = 1;
alert("bb");
</script>
</head>
<body>
</body>
<script>
alert("cc");
</script>
</html>

5、javascript输出脚本的执行顺序

在JavaScript开发中,经常会使用document对象的write()方法输出JavaScript脚本。document.write()方法先把输出的脚本字符串写入到脚本所在的文档位置,浏览器在解析完document.write()所在文档内容后,继续解析document.write()输出的内容,然后才按顺序解析后面的HTML文档。也就是说,JavaScript脚本输出的代码字符串会在输出后马上被执行。请注意,使用document.write()方法输出的JavaScript脚本字符串必须放在同时被输出的<script>标签中,否则JavaScript解释器因为不能够识别这些合法的JavaScript代码,而作为普通的字符串显示在页面文档中。但是,通过document.write()方法输出脚本并执行也存在一定的风险,因为不同JavaScript引擎对其执行顺序不同,同时不同浏览器在解析时也会出现Bug。

以上所述是小编给大家介绍的JavaScript语句的执行过程,希望对大家有所帮助。

Javascript 相关文章推荐
jQuery UI 实现email输入提示实例
Aug 15 Javascript
javascript校验价格合法性实例(必须输入2位小数)
May 05 Javascript
jQuery trigger()方法用法介绍
Jan 13 Javascript
javascript实现的网站访问量统计代码
Dec 20 Javascript
JS用斜率判断鼠标进入DIV四个方向的方法
Nov 07 Javascript
jQuery实现字符串全部替换的方法【推荐】
Mar 09 Javascript
jQuery实现动态删除LI的方法
May 30 jQuery
Express本地测试HTTPS的示例代码
Jun 06 Javascript
vue2.0 可折叠列表 v-for循环展示的实例
Sep 07 Javascript
php结合js实现多条件组合查询
May 28 Javascript
微信小程序 确认框的实现(附代码)
Jul 23 Javascript
使用JavaScript实现网页秒表功能(含开始、暂停、继续、重置功能)
Jun 05 Javascript
Javascript复制实例详解
Jan 28 #Javascript
基于jQuery实现以手风琴方式展开和折叠导航菜单
Jan 28 #Javascript
基于JavaScript的操作系统你听说过吗?
Jan 28 #Javascript
js+canvas绘制矩形的方法
Jan 28 #Javascript
js+canvas简单绘制圆圈的方法
Jan 28 #Javascript
谈一谈javascript闭包
Jan 28 #Javascript
JavaScript统计字符串中每个字符出现次数完整实例
Jan 28 #Javascript
You might like
Classes and Objects in PHP5-面向对象编程 [1]
2006/10/09 PHP
PHP 多维数组的排序问题 根据二维数组中某个项排序
2011/11/09 PHP
php绘图之在图片上写中文和英文的方法
2015/01/24 PHP
JavaScript中的Window窗口对象
2008/01/16 Javascript
js制作的鼠标悬浮时产生的下拉框效果
2012/10/27 Javascript
页面右下角弹出提示框示例代码js版
2013/08/02 Javascript
jquery禁止输入数字以外的字符的示例(纯数字验证码)
2014/04/10 Javascript
一个简单的jQuery计算器实现了连续计算功能
2014/07/21 Javascript
当前流行的JavaScript代码风格指南
2014/09/10 Javascript
jQuery中[attribute^=value]选择器用法实例
2014/12/31 Javascript
jQuery简单实现提交数据出现loading进度条的方法
2016/03/29 Javascript
JS在onclientclick里如何控制onclick的执行
2016/05/30 Javascript
解决bootstrap导航栏navbar在IE8上存在缺陷的方法
2016/07/01 Javascript
js控制文本框只能输入中文、英文、数字与指定特殊符号的实现代码
2016/09/09 Javascript
详解JavaScript权威指南之对象
2016/09/27 Javascript
js生成随机数方法和实例
2017/01/17 Javascript
微信小程序城市定位的实现实例(获取当前所在国家城市信息)
2017/05/17 Javascript
安装vue-cli报错 -4058 的解决方法
2017/10/19 Javascript
vue项目中v-model父子组件通信的实现详解
2017/12/10 Javascript
教你如何用node连接redis的示例代码
2018/07/12 Javascript
Laravel admin实现消息提醒、播放音频功能
2019/07/10 Javascript
layui清除radio的选中状态实例
2019/11/14 Javascript
使用PreloadJS加载图片资源的基础方法详解
2020/02/03 Javascript
Windows下安装 node 的版本控制工具 nvm
2020/02/06 Javascript
微信小程序向Java后台传输参数的方法实现
2020/12/10 Javascript
python图像和办公文档处理总结
2019/05/28 Python
Python Numpy计算各类距离的方法
2019/07/05 Python
.img/.hdr格式转.nii格式的操作
2020/07/01 Python
英国定做窗帘和纺织品面料一站式商店:Dekoria
2018/08/29 全球购物
Java的基础面试题附答案
2016/01/10 面试题
销售人员自我评价
2014/02/01 职场文书
审计专业自荐信范文
2014/04/21 职场文书
班风口号
2014/06/18 职场文书
立春观后感
2015/06/18 职场文书
幼儿园春季开学通知
2015/07/16 职场文书
深入探讨opencv图像矫正算法实战
2021/05/21 Python