资讯专栏INFORMATION COLUMN

你不知道的javascript(上卷)读后感(一)

zhaofeihao / 752人阅读

摘要:遮蔽效应作用域查找会在找到第一个匹配的标识符时停止,不会继续往上层作用域查找,这就会产生遮蔽效应。会发现每一次输出的都是为啥勒所有的回调函数回在循环结束后才会执行事件循环。

三剑客

编译,顾名思义,就是源代码执行前会经历的过程,分三个步骤,

分词/词法分析,将我们写的代码字符串分解成多个词法单元

解析/语法分析,将词法单元集合生成抽象语法树(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成员是否存在。

原文地址

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/106384.html

相关文章

  • 你不知道javascript(上卷)后感(二)

    摘要:词法熟悉语法的开发者,箭头函数在涉及绑定时的行为和普通函数的行为完全不一致。被忽略的作为的绑定对象传入,使用的是默认绑定规则。使用内置遍历数组返回迭代器函数普通对象不含有,无法使用,可以进行改造,个人博客地址 this词法 熟悉ES6语法的开发者,箭头函数在涉及this绑定时的行为和普通函数的行为完全不一致。跟普通this绑定规则不一样,它使用了当前的词法作用域覆盖了this本来的值。...

    Ali_ 评论0 收藏0
  • 十分钟快速了解《你不知道 JavaScript》(上卷

    摘要:最近刚刚看完了你不知道的上卷,对有了更进一步的了解。你不知道的上卷由两部分组成,第一部分是作用域和闭包,第二部分是和对象原型。附录词法这一章并没有说明机制,只是介绍了中的箭头函数引入的行为词法。第章混合对象类类理论类的机制类的继承混入。 最近刚刚看完了《你不知道的 JavaScript》上卷,对 JavaScript 有了更进一步的了解。 《你不知道的 JavaScript》上卷由两部...

    赵春朋 评论0 收藏0
  • 你不知道JavaScript上卷之作用域与闭包·读书笔记

    摘要:的分句会创建一个块作用域,其声明的变量仅在中有效。而闭包的神奇作用是阻止此事发生。依然持有对该作用域的引用,而这个引用就叫做闭包。当然,无论使用何种方式对函数类型的值进行传递,当函数在别处被调用时都可以观察到闭包。 date: 16.12.8 Thursday 第一章 作用域是什么 LHS:赋值操作的目标是谁? 比如: a = 2; RHS:谁是赋值操作的源头? 比如: conso...

    Raaabbit 评论0 收藏0
  • 重读你不知道JS (上) 第节三章

    摘要:如果是声明中的第一个词,那么就是一个函数声明,否则就是一个函数表达式。给函数表达式指定一个函数名可以有效的解决以上问题。始终给函数表达式命名是一个最佳实践。也有开发者干脆关闭了静态检查工具对重复变量名的检查。 你不知道的JS(上卷)笔记 你不知道的 JavaScript JavaScript 既是一门充满吸引力、简单易用的语言,又是一门具有许多复杂微妙技术的语言,即使是经验丰富的 Ja...

    lavor 评论0 收藏0
  • 重读你不知道JS (上) 第节五章

    摘要:词法作用域的查找规则是闭包的一部分。因此的确同闭包息息相关,即使本身并不会真的使用闭包。而上面的创建一个闭包,本质上这是将一个块转换成一个可以被关闭的作用域。结合块级作用域与闭包模块这个模式在中被称为模块。 你不知道的JS(上卷)笔记 你不知道的 JavaScript JavaScript 既是一门充满吸引力、简单易用的语言,又是一门具有许多复杂微妙技术的语言,即使是经验丰富的 Jav...

    worldligang 评论0 收藏0

发表评论

0条评论

zhaofeihao

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<