你不知道的javascript(上卷)读后感(一)
三剑客
编译,顾名思义,就是源代码执行前会经历的过程,分三个步骤,
- 分词/词法分析,将我们写的代码字符串分解成多个词法单元
- 解析/语法分析,将词法单元集合生成抽象语法树(AST)
- 代码生成,抽象语法树(AST)转换成可执行代码的过程
Tip1:js在语法分析和代码生成阶段有对运行性能进行优化,对冗余元素进行优化
Tip2:js的编译过程不是发生在构建之前,而是代码执行之前
理解作用域,首先知道三剑客,分别是
- 引擎:负责整个代码编译及执行的过程
- 编译器: 负责词法分析、语法分析、代码生成
- 作用域:负责维护与收集所有声明的标识符,保证当前执行代码对这些标识符的访问权限
举例子,加深印象,对于var a = 2,三剑客如何协同工作,
编译器进行分词、语法分析,然后要代码生成时,遇到 var a,问一下当前作用域集合“你有没有这个名称的变量呀?”,作用域如果说有,那么忽略声明,继续编译,如果说没有,那么就要要求作用域收集一下,并且给它个名字a,然后编译器就生成了代码,引擎准备来执行了,先问当前作用域集合,“你这里有a这个变量吗?”,有引擎就拿来用,没有就继续找该变量,要么找到,就给它附个值2,没有那就给你报个错!
理解编译器的相关术语
- LSH查询,通俗解释就是找到所声明变量,并且对其赋值的行为
- RSH查询,通俗解释就是查找声明的变量
作用域嵌套
当一个块或是函数嵌套在另外一个块或函数时,就会产生作用域嵌套,于是在当前作用域找不到某个变量时,引擎会往外层嵌套作用域继续查找,直达到最外层作用域(全局作用域)为止,也就是所谓的作用域链啦!
词法作用域
相信有很多人都是搞不懂词法作用域是什么?
所谓的词法作用域,就是定义在词法阶段(词法分析)的作用域,由你写代码时将变量和块作用域写在哪里来决定的。
遮蔽效应
作用域查找会在找到第一个匹配的标识符时停止,不会继续往上层作用域查找,这就会产生遮蔽效应。
欺骗词法作用域
- eval函数,修改词法作用域
- with关键字,创建词法作用域
导致性能下降的原因,前面我们提过,在解析/语法分析、生成代码阶段,我们会对代码进行优化,剔除冗余元素,但是当使用eval/with时,所有优化变得没有意义,因为存在不可预见性,不知道修改和创建的词法作用域是什么?所以会导致性能下降。
函数声明、函数表达式、匿名函数表达式
函数表达式:function为第一个词,那么就是一个函数声明,否则就是一个函数表达式
匿名函数表达式:没有函数名,匿名函数在栈追踪这种不会显示有意义的函数名,使得调试困难
IIFE: 立即执行函数表达式((function() { ... }())、(function(){ ... })())
块级作用域
let为其声明的变量隐式地去劫持了所在的块级作用域,不会在块级作用域中进行提升【变量提升】
Demo: with关键字为块级作用域、{...}为块级作用域,用完即销毁
const常量,不可修改!
任何声明在某个作用域(函数作用域和块级作用域)的变量,都是属于这个作用域。
每个作用域都会进行提升操作。
函数声明会被提升,函数表达式不会提升,变量声明提升的过程中,函数会优先!
闭包
闭包,有权访问另外一个函数的变量标识符的函数,比较常见的一个闭包问题,就是for循环。
for(var i = 1; i <= 5; i++) {
setTimeout(function() {
console.log(i);
}, i*1000)
}
会发现每一次输出的都是6,为啥勒?所有的回调函数回在循环结束后才会执行(事件循环)。所以每次都是输出一个6来。
解决办法:
1、IIFE,每个迭代储存i的值
for(var i = 1; i <= 5; i++) {
(function(i) {
var j = i;
setTimeout(function() {
console.log(j);
}, j*1000);
})(i)
}
2、IIFE,将i入参修改成j
for(var i = 1; i <= 5; i++) {
(function(j) {
setTimeout(function() {
console.log(j);
}, j*1000);
})(i)
}
3、创建闭包的块作用域
for(var i = 1; i <= 5; i++) {
let j = i;
setTimeout(function() {
console.log(j);
}, j*1000);
}
4、最优
for(let i = 1; i <= 5; i++) {
setTimeout(function() {
console.log(i);
}, i*1000)
}
另外一个闭包常用的场景,就是模块暴露了,在这里提供一个现代模块机制的实现方式,大家可以细细品尝,
var MyMoudles = (function Manger() {
var modules = {};
function define(name, deps, impl) {
for (var i = 0; i < deps.length; i++) {
deps[i] = modules[deps[i]];
}
modules[name] = impl.apply(impl, deps);
}
function get(name) {
return modules[name];
}
})()
调用Demo:
MyMoudles.define('bar', [], function() {
function hello(who) {
return "Let me introduce: " + who;
}
return {
hello: hello
}
})
MyMoudles.define('foo', ['bar'], function(bar) {
var hungry = 'hippo';
function awesome() {
console.log(bar.hello(hungry).toUpperCase());
}
return {
awesome: awesome
}
})
var bar = MyMoudles.get('bar');
var foo = MyMoudles.get('foo');
console.log(bar.hello('hippo')); // Let me introduce: hippo
foo.awesome();
未来模块机制
ES6提供了全新的模块机制,基于函数的模块(如上述现代模块机制)并不是一个能被静态识别的模式(编译器无法识别),它们的API语义只有等到代码运行时才会考虑进来,而ES6模块就是一个能被静态识别的模式,就是说API在编译阶段就会检查API成员是否存在。
你不知道的javascript(上卷)读后感(一)的更多相关文章
- 你不知道的javascript(上卷)读后感(二)
this词法 熟悉ES6语法的开发者,箭头函数在涉及this绑定时的行为和普通函数的行为完全不一致.跟普通this绑定规则不一样,它使用了当前的词法作用域覆盖了this本来的值. 误解 this理解成 ...
- 你不知道的JavaScript上卷笔记
你不知道的JavaScript上卷笔记 前言 You don't know JavaScript是github上一个系列文章 初看到这一标题的时候,感觉怎么老外也搞标题党,用这种冲突性比较强的题目 ...
- 读《你不知道的JavaScript(上卷)》后感-作用域闭包(二)
github原文 一. 序言 最近我在读一本书:<你不知道的JavaScript>,这书分为上中卷,内容非常丰富,认真细读,能学到非常多JavaScript的知识点,希望广大的前端同胞们, ...
- 你不知道的javaScript上卷(第一章 作用域是什么)
在写这篇博客时这本书我已经是看过一遍了,为了加深印象和深入学习于是打算做这系列的前端经典书籍导读博文,大家如果觉得这本书讲的好可以自己买来看看,我是比较喜欢看纸质版书的,因为这样才有读书的那种感觉. ...
- 读《你不知道的JavaScript(上卷)》后感-浅谈JavaScript作用域(一)
原文 一. 序言 最近我在读一本书:<你不知道的JavaScript>,这书分为上中卷,内容非常丰富,认真细读,能学到非常多JavaScript的知识点,希望广大的前端同胞们,也入手看看这 ...
- 《你不知道的 JavaScript 上卷》 学习笔记
第一部分: 作用域和闭包 一.作用域 1. 作用域:存储变量并且查找变量的规则 2. 源代码在执行之前(编译)会经历三个步骤: 分词/此法分析:将代码字符串分解成有意义的代码块(词法单元) 解析/语法 ...
- 【你不知道的javaScript 上卷 笔记3】javaScript中的声明提升表现
console.log( a ); var a = 2; 执行输出undefined a = 2; var a; console.log( a ); 执行输出2 说明:javaScript 运行时在编 ...
- JS闭包—你不知道的JavaScript上卷读书笔记(二)
关于闭包,初学者会被绕的晕头转向,在学习的路上也付出了很多精力来理解. 让我们一起来揭开闭包神秘的面纱. 闭包晦涩的定义 看过很多关于闭包的定义,很多讲的云里雾里,晦涩难懂.让不少人以为闭包是多么玄乎 ...
- JavaScript词法作用域—你不知道的JavaScript上卷读书笔记(一)
前段时间在每天往返的地铁上抽空将 <你不知道的JavaScript(上卷)>读了一遍,这本书很多部分写的很是精妙,对于接触前端时间不太久的人来说,就好像是叩开了JavaScript的另一扇 ...
随机推荐
- Angular实现简单数据计算与删除
AngularJS 1)什么是AngularJS AngularJS 简介 AngularJS 是一个 JavaScript 框架.它可通过 <script> 标签添加到 HTML 页面. ...
- 123457123457#0#-----com.yuming.YiZhiFanPai01--前拼后广--益智早教游戏记忆翻牌cym
com.yuming.YiZhiFanPai01--前拼后广--益智早教游戏记忆翻牌cym
- WinSCP 上传文件至Cetos 7 后用户无权限
WinSCP是一个支持SSH的SCP文件传输软件. 可以用Windows环境向Linux环境传输文件,今天给新的Elasticsearch 服务器(cetos 7 )部署新的集群节点的时候,发现传输后 ...
- 【Leetcode_easy】788. Rotated Digits
problem 788. Rotated Digits solution1: class Solution { public: int rotatedDigits(int N) { ; ; i< ...
- iOS开发设计模式
ios开发学习中,经常弄不清楚ios的开发模式,今天我们就来进行简单的总结和探讨~ (一)代理模式 应用场景:当一个类的某些功能需要由别的类来实现,但是又不确定具体会是哪个类实现. 优势:解耦合 敏捷 ...
- Win10利用CodeBlocks搭建Objective-C开发环境(二)
工程文件已经建好:但此时会发现main.m文件为灰色,且无法点击,此时需右键点击main.m文件,在option选项中勾选 compile file和 link file选项. 设置完成后,双击mai ...
- webpack简单配置
1.代理配置 需要修改一下配置文件 config里的index.js,根据接口特点自主选取 2.解决图标显示路径错误问题 项目在打包完成后如果出现图片显示不了的问题,需要进行如下配置
- 【ARM-Linux开发】Linux查看设备驱动
驱动操作命令: insmod / modprobe 加载驱动 rmmod 卸载驱动 lsmod 查看系统中所有已经被 ...
- Go之接口interface(1)
1. 什么是interface在此之前,我们遇到的都是具体的类型,比如数字类型.切片类型等等.对于这些具体的类型,我们总是能知道它是什么.可以利用它来做什么,比如对于一个数字类型,我们知道可以对其进行 ...
- web 中常用的两种上传文件的方法总结
这里我们来总结整理一下常用的两种文件上传方式以及要注意的东西: 1.springmvc .MultipartFile 的上传方式. 2.org.apache.commons.fileupload 使用 ...