You Don't Know JS: Scope & Closures (第一章:什么是Scope)
Content
- What is Scope?
- Lexical Scope
- Function Vs. Block Scope
- Hoisting
- Scope Closures
Appendix:
- Dynamic Scope
- Polyfilling Block Scope
- Lexical-this
- Thank You's
Chapter1: What is Scope?
In fact, the ability to store values and pull values out of variables is what gives a program state.
背后的问题:
变量储存在哪?
当需要用到它们,程序怎样找到它们?
回答:
需要a well-defined set of rules: Scope.
但是Scope在哪里和如何得到设置get set?
Compiler Theory
JS是编译后的语言。
Compilation的几个大步骤:
- Tokenizing/Lexing(代币/类型分析): 把a string of characters分解为有意义的小块,被成为tokens。
- Parsing: 把tokens转化进一个嵌套元素的tree结构。集体地代表程序的语法结构。
- tree叫做Abstract Syntax Tree(提取句法树)
var a = 2;
//被分解成:
/*
顶层: VariableDeclaration
子节点 Identifier: a
子节点 AssignmentExpression: =
子节点 NumericLiteral: 2
*/
- Code-Generation: 把这个AST转化为可执行的代码。(根据不同语言,有区别)。
- 转化为机器指令,目的是来创建一个变量a, 并储存一个value到a.
当然JS引擎更复杂。编译时间,在许多情况下,仅仅耗时微秒或更少。并立即在编译后执行代码。
Understanding Scope
学习理解Scope,可以比喻成,进程的一场谈话。
var a = '123'
The Cast 全体演员
- Engine 响应从开始到结束编译,并执行JS程序。
- Compiler 编译器。处理所有脏活,包括parsing and code-generation.
- Scope 收集和保存一个名单。名单上包括所有的变量identifiers,并强迫实施一个严格的规则:这些identifiers如何存取到 currently executing code。
Back & Forth
Engine看2个独立的声明。1个是在编译期间,编译器处理。另一个是在执行期间,引擎处理。
首先,编译器Tokening and Lexing,
然后, Parsing 代码到AST树,
最后, Code-Generation。(用到Scope和Engine的交互。)
- 编译器询问Scope"变量a是否存在于你的list中?"。
- 如果是,继续; 如果否,编译器要求Scope去声明一个新的变量a
- 编译器生产代码用于Engine在后续执行,去处理a = 2。
- Engine 跑起来,先询问Scope:你的当前list里面有变量a吗?
- 如果有,Engine使用这个变量, assignment.
- 如果没有,Engine去其他地方查看(查看嵌套的Scope部分)
总结:
编译器先声明变量,然后Engine在Scope中查询这个变量,如果发现就分配它。
首先Compiler Speak
然后Engine/Scope Conversation
Engine执行代码,是在编译器生产后的第2步,Engine查询变量a是否已被声明。这个查询是咨询Scope。
Engine执行的查询的类型(LHS和RHS),影响查询的结果outcome。
本例子:var a = '123';
Engine将执行一个“Left-hand Side”查询,查询内容是变量a。
还有"Right-hand Side",准确的描述是:非左侧查询。
Side是指分配符号an assignment operation的左右两边。
如: console.log(a); 就是非左侧查询。
其实LHS和RHS只是涉及是否assign的一种称呼。
"who's the target of the assignment (LHS)" and "who's the source of the assignment (RHS)".
(这两个概念有什么用??见后,ES6以后区别不明显了。)
Engine和Scope的对话:
function foo(a) {
console.log( a ); //
} foo( 2 );
Engine: Scope,我有一个右侧查询, 引用foo,听说过它吗?
Scope: 当然,我有。Compiler在一秒前声明了它。它是函数。给你!
Engine: 太好了,
You Don't Know JS: Scope & Closures (第一章:什么是Scope)的更多相关文章
- Node.js学习(第一章:Node.js安装和模块化理解)
Node.js安装和简单使用 安装方法 简单的安装方式是直接官网下载,然后本地安装即可.官网地址:nodejs.org Windows系统下,选择和系统版本匹配的.msi后缀的安装文件.Mac OS ...
- Node.js学习(第一章:Node.js简介)
Node.js是什么? Node.js 诞生于 2009 年,由 Joyent 的员工 Ryan Dahl 开发而成, 目前官网最新版本已经更新到 12.0.0版本,最新稳定的是10.15.3.Nod ...
- You Don't Know JS: Scope & Closures(翻译)
Chapter 1: What is Scope? 第一章:什么是作用域 One of the most fundamental paradigms of nearly all programming ...
- You Don't Know JS: Scope & Closures (第3章: 函数 vs 块作用域)
第二章,作用域由一系列的bubbles组成.每一个都代表了一个container或bucket,装着被声明的identifiers(variables, functions).这些bubbles相互嵌 ...
- You Don't Know JS: Scope & Closures (第2章: Lexical Scope)
2种主要的models for how scope work. 最普遍的是Lexical Scope. 另一种 Dynamic Scope.(在Appendix a中介绍.和Lexical Scope ...
- You Don't Know JS: Scope & Closures (附加:Lexical/dynamic作用域)(附加:Lexical-this)
JavaScript只有Lexical Scope 模式 Lexical Scope就是在写代码的时候,定义函数的时候创建的作用域! 而动态作用域是在runtime时,函数被调用的地方的作用域! 实际 ...
- 【vue.js权威指南】读书笔记(第一章)
最近在读新书<vue.js权威指南>,一边读,一边把笔记整理下来,方便自己以后温故知新,也希望能把自己的读书心得分享给大家. [第1章:遇见vue.js] vue.js是什么? vue.j ...
- 第一章 用three.js创建你的第一个3D场景
第一章 用three.js创建你的第一个3D场景 到官网下载three.js的源码和示例. 创建HTML框架界面 第一个示例的代码如下: 01-basic-skeleton.html 位于 Learn ...
- 【js 笔记】读阮一峰老师 es6 入门笔记 —— 第一章
鉴于最近用 vuejs 框架开发项目,其中有很多涉及到 es6 语法不太理解所以便认真地读了一下这本书. 地址:http://es6.ruanyifeng.com/#README 第一章:let ...
随机推荐
- centos设置中文输入法无效的解决办法
安装 im-chooser: yum install im-chooser 回到当前使用的普通用户,设置 ibus 输入法为默认输入系统: imsettings-switch ibus
- 二次剩余&&Cipolla
目录 二次剩余 勒让德符号(legendre symbol) Cipolla's Algorithm. 代码 end 二次剩余 给定y和奇质数p,求x,使得\(x^2≡y(mod p)\) 勒让德符号 ...
- codevs1017乘积最大
codevs1017 乘积最大 题目描述 Description 今年是国际数学联盟确定的“2000——世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场 ...
- POJ-1038 Bugs Integrated, Inc. (状压+滚动数组+深搜 的动态规划)
本题的题眼很明显,N (1 <= N <= 150), M (1 <= M <= 10),摆明了是想让你用状态压缩dp. 整个思路如下:由于要填2*3或者3*2的芯片,那么就要 ...
- Java基础【冒泡、选择排序、二分查找】
冒泡排序的思路就是前一个和后一个进行比较,如果大的就交换位置 大的数字后浮 如 12 8 5 31 第一轮 8 5 12 31 第二轮 5 8 ...
- 【面试问题】mybatis 与 Hibernate的不同
Mybatis和hibernate不同,它不完全是一个ORM框架,因为MyBatis需要程序员自己编写Sql语句.mybatis可以通过XML或注解方式灵活配置要运行的sql语句,并将java对象和s ...
- JDBC编程的步骤
一.进行JDBC编程的步骤大致如下: 1. 加载数据库驱动,通常使用Class类的forName()静态方法来加载驱动.如下代码: Class.forName(dirvirClass) 上面 ...
- SpringMVC统一转换null值为空字符串的方法 !
在SpringMVC中,可以通过在<mvc:annotation-driven>中配置<mvc:message-converters>,把null值统一转换为空字符串,解决这个 ...
- Java 静态方法不能重写但可以被子类静态方法覆盖
强调 静态方法是属于类的,只存在一份,会被该类的所有对象共享.不可以被重写. 静态方法可以被子类继承,但是不可以被子类重写 class door{ } class wood_Door extends ...
- AngularJS 笔记1
2017-03-23 本文更新链接: http://www.cnblogs.com/daysme/p/6606805.html 什么是 angularjs 2009年有两个外国人创建,后由谷歌收购并开 ...