javascript代码是如何执行的呢,分为六个步骤(就像把大象装进冰箱总共分几步?):

  第一步:载入第一个js代码段(注:script标签对内的代码或是引用js代码,这也说明js并不是一行一行(单纯意义上的自上而下)执行的,而是一段一段执行的)。

  第二步:词法分析、语法分析,如果这时有语法错误,解释器便会终止执行该代码并抛出语法(Syntax Error)的错误,并转达到第五步。

  第三步:对段内的var和function做预解析,这一步不会报错。

  第四步:执行代码。

  第五步:如果还有代码段,则载入下个代码,跳到第二步接着来。

  第六步:结束。

流程图为:

  首先:在js解释器启动时或web浏览器加载新的页面时,会自动创建一个全局对象,并给它定义一组初始属性,在客户端javascript中,window便是这个全局对象,它有一个window属性引用自身,代替this来引用全局对象;如果代码中声明了一个全局变量,那么这个全局变量便是全局对象中的一个属性(查看方式为在firefox或chrome中,按F12键调出控制台,在控制台中输入for(var i in window) {console.log(i + ' ' + window[i])},便可看到初始属性)。全局域(window)下所有的js代码,可以看成一个被自动执行的“匿名方法”,而“匿名方法”内的方法,则需要显示调用才被执行。例如:

(function () {
    function a() {
        console.log('我是a方法,需要显示调用才执行');
    }
    console.log('我是匿名方法内的,被自动执行了');
    a();
})()  

其次:上述流程概括就是解析和执行两个阶段。

  第一阶段:通过词法分析、预解析生成语法分析树。

  第二阶段:执行。执行某个具体的function时,js解释器会为它创建一个执行环境(ExecutionContext)和活动对象(ActiveObject)。

之前描述过了词法分析的流程,这里就不过多的分析了,直接上图:

下面,直接上题:

if(!('a' in window)){
    var a = 1;
}
alert(a); // undefined;

有的同学就会感到奇怪了,为什么啊,为什么不是1呢,也没见到其他地方声明a啊,所以a属性不在全局域内啊,理应弹出1啊。别急,且听我细细解释。

首先看看上面列出的六个步骤,在第三步js解释器会对代码中的function和var进行预解析,也就是说在碰到var声明时,会把var声明提到顶部,于是上段代码便成了这样

var a;
if(!('a' in window)){
    a = 1;
}
alert(a); // undefined;

关于变量的声明的步骤之前文章中就讲过了,这里就不多说了。

看到这儿,同学们应该不难理解为什么弹出undefined了吧。需要说明的是在js中,对于var a=1;这样代码,其实是分为两步执行的,先声明,再赋值,如果没有赋值,那么a的值便是undefined。

再来第二题:

var a = 1,
    b = function (x) {
        x && b(--x);
    };
alert(a); // 1

这题看上去便比较纠结了吧,定义了个a变量,接着又定义了个有名函数a,这都什么和什么啊。其实完全不必纠结,我改变下写法,同学们便会明白

var a = 1,
    b = function (x) {
        x && b(--x);
    };
alert(a);  

怎么样,再来一题:

function a(x) {
    return x * 2;
}
var a;
alert(a); 

认真看完第一题的同学,会有部分说,这有何难,不就是弹出undefined么,可结果是,真的是那样的么?结果是

function a(x) {
    return x * 2;
}  

  啊,怎么会这样呢,不是说var a,只声明没定义,就是undefined么,为什么会弹出一个函数呢。别急,还是第三步。js解释器会对代码中的function和var进行预解析,那么如果两者同时存在呢,而且都是对同一个a进行声明,那又如何处理?事实上,对于这种情况,js解释器早已考虑到了,当两者同时存在时,函数声明的优先级大于变量声明,而且如果变量只声明而没有赋值的话,便会覆盖它。记住,如果变量声明的同时,也赋值了,那么就不一样了,如:

function a(x) {
    return x * 2;
}
var a = 1;
alert(a);  

这时便会弹出1了。我们再深入点,看看下面的代码:

function a(x) {
    return x * 2;
}
var a = 1;
alert(typeof a);  

这时会弹出number,对,这时函数声明便会被变量声明覆盖。

还没完,再来一题:

function b(x, y, a) {
    arguments[2] = 10;
    alert(a);
}
b(1, 2, 3);  

这题主要是考察了方法内的arguments对象,这个是类数组的对象,真实记录了方法的形参个数和长度,但记住,它不是数组,只是个长的像数组的对象

这个问题在之前的文章里也谈过了。

var argumengts = {0:1, 1:2, 2:3, length:3}  

看到这,结果弹出什么就一目了然了吧。

还有一题:

function a() {
    alert(this);
}
a.call(null);  

  这题我觉得主要考察了两个知识点,this的指向谁和call如何把this强制改变并指向谁。在上面的几个关键概念中,解释了js词法作用域是在定义时决定而不是执行时决定,因此可以静态分析。那么我们就来分析下,撇开下面的a.call(null),方法a是在全局下定义的,因此在全局作用域下调用a,this便会指向调用它的那个对象window,于是

function a() {
    alert(this); // 指向window
}
a() // 在全局域内等同于window.a()  

那么a.call(null)呢,this指向null,那么弹出什么呢?在ECMAScript262中,如果call,apply方法中第一个参数传入null,等同于传入window,因此和上述代码一样。 

javascript --- 再谈词法分析的更多相关文章

  1. JavaScript 再谈闭包

    之前有整理过一版关于闭包的概念,但感觉思路不是很清晰,是临时想起一些例子来讲的,今天再次来讲一下闭包. 闭包: 函数嵌套函数,内部函数可以引用外部函数的参数和变量 function aaa(a){ v ...

  2. 再谈JavaScript的数据类型问题

    JavaScript的数据类型问题已经讨论过很多次了,但许多人还有许多书仍然沿用着错误的.混乱的一些观点,所以就再细讲一回. 提及这个讨论的原因在于argb同学在我的MSN博客上的一段回复,又更早的起 ...

  3. 再谈JSON -json定义及数据类型

    再谈json 近期在项目中使用到了highcharts ,highstock做了一些统计分析.使用jQuery ajax那就不得不使用json, 可是在使用过程中也出现了非常多的疑惑,比方说,什么情况 ...

  4. 再谈angularJS数据绑定机制及背后原理—angularJS常见问题总结

    这篇是对angularJS的一些疑点回顾,是对目前angularJS开发的各种常见问题的整理汇总.如果对文中的题目全部了然于胸,觉得对整个angular框架应该掌握的七七八八了.希望志同道合的通知补充 ...

  5. 再谈前端HTML模板技术

    在web2.0之前,写jsp的时候虽然有es和JSTL,但是还是坚持jsp.后面在外包公司为了快速交货,还是用了php Smart技术. web2.0后,前端模板技术风行. 代表有如下三大类: Str ...

  6. 再谈DOMContentLoaded与渲染阻塞—分析html页面事件与资源加载

    浏览器的多线程中,有的线程负责加载资源,有的线程负责执行脚本,有的线程负责渲染界面,有的线程负责轮询.监听用户事件. 这些线程,根据浏览器自身特点以及web标准等等,有的会被浏览器特意的阻塞.两个很明 ...

  7. 再谈HTTP2性能提升之背后原理—HTTP2历史解剖

    即使千辛万苦,还是把网站升级到http2了,遇坑如<phpcms v9站http升级到https加http2遇到到坑>. 因为理论相比于 HTTP 1.x ,在同时兼容 HTTP/1.1 ...

  8. 再谈 Go 语言在前端的应用前景

    12 月 23 日,七牛云 CEO & ECUG 社区发起人许式伟先生在 ECUG Con 2018 现场为大家带来了主题为<再谈 Go 语言在前端的应用前景>的内容分享. 本文是 ...

  9. 再谈js对象数据结构底层实现原理-object array map set

    如果有java基础的同学,可以回顾下<再谈Java数据结构—分析底层实现与应用注意事项>:java把内存分两种:一种是栈内存,另一种是堆内存.基本类型(即int,short,long,by ...

随机推荐

  1. 解决删除域用户Exception from HRESULT: 0x80072030

    解决删除域用户异常问题. System.DirectoryServices.DirectoryServicesCOMException was unhandled  Message=在服务器上没有这样 ...

  2. 在Java中直接调用js代码(转载)

    http://blog.csdn.net/xzyxuanyuan/article/details/8062887 JDK1.6版添加了新的ScriptEngine类,允许用户直接执行js代码. 在Ja ...

  3. [原]Android Native Debug

    1,安装adt插件,cdt插件2,SDK目录配置: Eclipse文件菜单选择“Window”--->“Preferences”--->“Android”--->设置“SDK Loc ...

  4. C#简易播放器(基于开源VLC)

    可见光通信技术(Visible Light Communication,VLC)是指利用可见光波段的光作为信息载体,不使用光纤等有线信道的传输介质,而在空气中直接传输光信号的通信方式.LED可见光通信 ...

  5. Qt Creator 黑色主题配置

    可能是一个习惯了吧,我个人比较喜欢在黑色主题的环境下进行编程.黑色主题对眼睛稍微友好一点,看起来也不是那么low.这里给出QtCreator的黑色主题配置方案. 如果是最新的Creator3.3+的版 ...

  6. 基于jQuery HTML5人物介绍卡片特效

    基于jQuery HTML5人物介绍卡片特效.这是一款基于jquery.material-cards插件实现的人物介绍卡片形式特效代码.效果图如下: 在线预览   源码下载 实现的代码. html代码 ...

  7. php -- 获取当月天数及当月第一天及最后一天、上月第一天及最后一天(备忘)

    Learn From :http://www.jxbh.cn/newshow.asp?id=1635&tag=2 //1.获取上个月第一天及最后一天. date('Y-m-01', strto ...

  8. NPM使用详解(上)

    1.NPM是什么? NPM是JavaScript的包管理工具,在安装NodeJS(什么?你不知道node?来,我们合计合计:https://nodejs.org/)的时候,会自动安装上npm. 要查看 ...

  9. Screensiz.es – 最流行移动设备及显示器的屏幕规格大全

    Screensiz.es 帮助您快速找到目前市场上最流行的设备和显示器的屏幕规格.尺寸数据来自维基百科,使用更好理解的像素密度.流行度推算自 Google 查询(从 AdWords 流量估算),以及一 ...

  10. node debug包

    'use strict'; var debug = require('debug')('http') , http = require('http') , name = 'My App'; // fa ...