关于《你不知道的javascript》
编译原理
关于编程语言,我们会想到的一个基本知识点是:首先它可以存储变量,其次它能够对这个变量进行访问与修改。对于javascript,实现这些功能多于其编译器、引擎与作用域有关。那么我们今天就先从编译器简单说起吧。
对于一般的编译语言,编译过程多在编码前期执行,通过分词==语法分析==代码生成来将代码编译成机器指令。而对于js来说,编译的过程也是比较复杂的,其中还涉及到很多的优化项目,但是一般来说,它没有那么多的编译时间,只会在执行前对作用域中的代码进行编译。
工作机制
以var a=2举个例子吧,介绍一下编译器与引擎和作用域怎样工作的
- 首先,编译器看到了var a,向作用域来问a是否声明过,是,则无视,继续生成后面的代码,否,向作用域中添加a这个变量。后续以代码生成方式,生成a=2这个代码,但没有什么实际意义。
- 引擎开始工作,对于a=2的赋值语句,先向作用域查找a变量,确认a存在,将2赋值于a,如果这里不存在,则会举爪报错。
这里我们可以看到编译器与引擎都会与作用域有相应的交集,但是因为编辑器只需要判断是否需声明以及作用域声明吧这样的作用,好像体现不了作用域的重要性,作者就不说他两之间的事了,转向一个更重要的地方,引擎如何在作用域中查找变量,以及确认其是在哪一层--好像确实重要些,其实就是不好理解嘛,就找了两个方法来帮助我们去理解一下这样的查找机制。
那些赋值左右的事情
我们看到的表达式,其实大多只有两种,一种是直接有‘=’的赋值语句,一种是纯粹的函数执行:console.log(‘就酱’)。
- 那么对于赋值语句a=b来说,对于a的查询,我们可以知道,它只需要将告诉引擎==‘我就在这,我的值你添加就好啦’,但是对于b,我们需要先找到有没有b,即是否声明过,没有声明,抱歉,你出现了ReferenceError错误啦。同样,如果是console.log(b),也是先查找到源头,看是否声明。
- 这里可以看出我们大致可以将在作用域中的查找变成两种思路:LHS查询和RHS查询,可以记忆为赋值左右的小窍门--为什么这么叫就不造啦。左边的是LHS,a=。。。,我们只需要知道a存在,能找到它-即赋值的目标。右边的RHS,a=b中的b可以理解为赋值的源头,要确定它是否被声明,是否有值。
RHS与LHS的意义何在
- 两种查询思路,更方面我们知道变量在作用域中是如何被查找到的
- 区别:非严格模式下,a=2没有声明a一点毛病没有,但是a=b,b没有声明就引用错误啦;而且如果var b,a;a.b这种的b也是不行哒,会说类型错误==TypeError
- 然后就是对于多层嵌套的作用域,我们可以很清楚的用这两种方式去找变量啦,不着急。
作用域是什么
说了半天,那么作用域到底是什么,书里那一句话还不错:作用域就是一种方法,用于确定在何处以及如何查找变量。
小罗嗦
写文好慢的速度,那就随便说说啦,书中的观点自己的话,有问题大家评论呀,一起讨论讨论