你不知道的JavaScript(作用域和闭包)

时间:2023-02-23 13:00:01

作用域和闭包

・作用域

  引擎:从头到尾负责整个JavaScript的编译及执行过程。

  编译器:负责语法分析及代码生成等。

  作用域:负责收集并维护由所有声明的标识符(变量)组成的一系列查询,并实施一套非常严格的规则,确定当前执行的代码对这些标识符的访问权限。

  作用域是一套规则,用于确定在何处以及如何查找变量(标识符)。

  如果查找的目的是对变量进行赋值,那么就会使用LHS查询;

  如果目的是获取变量的值,就会使用RHS查询。

・词法作用域

  无论函数在哪里被调用,也无论它如何被调用,它的词法作用域都只由函数被声明时所处的位置决定。

  欺骗词法:两种机制。欺骗词法作用域会导致性能下降。

  eval():

  function foo(str, a) {

    eval(str);

    console.log(a, b);

  }

  var b = 2;

  foo(“var b = 3; ”, 1);

  // 1, 3

  with():

  function foo(obj) {

    with(obj) {

  a = 2;

  }

 }

  var o1 = { a: 3};

  var o2 = { b: 3};

  foo(o1);

  console.log(o1.a); // 2

  foo(o2);

  console.log(o2.a); // undefined

  console.log(a); // 2

・函数作用域和块作用域

  隐藏内部实现:可以把变量和函数包裹在一个函数的作用域中,然后用这个作用域来隐藏它们。

  最小特权原则(最小授权或最小暴露原则):最小限度地暴露必要内容,而将其他内容都隐藏起来。

  规避冲突:隐藏作用域中的变量和函数,可以避免同名标识符之间的冲突。

  包装函数:

  (function foo() {

    var a = 3;

    console.log(a);

  })();

  匿名函数:

  setTimeout(function() {…}, 1000);

  缺点:

  匿名函数在栈追踪中不会显示出有意义的函数名,使得调试困难;

  如果没有函数名,当需要引用自身时只能使用已过期的arguments.callee引用;

  匿名函数省略了对于代码可读性/可理解性很重要的函数名。

  给函数表达式指定一个函数名可以有效解决以上问题:

  setTimeout(function timeoutHandler() {…}, 1000);

  块作用域:

    let关键字可以将变量绑定到所在的任意作用域中;

    使用let进行的声明不会在块作用域中进行提升;

  提升:

    只有声明本身会被提升,而赋值或其他运行逻辑会留在原地。

    如果提升改变了代码执行的顺序,会造成非常严重的破坏;

  函数声明和变量声明都会被提升,但是函数会首先被提升;

  foo(); // 1

  var foo;

  function foo() {

console.log(1);

  }

  foo = function() {

console.log(2);

  }

  ・作用域闭包

  只要使用了回调函数,实际上就是在使用闭包;

  模块模式具备的两个必要条件:

  必须有外部的封闭函数,该函数必须至少被调用一次;

  封闭函数必须返回至少一个内部函数,这样内部函数才能在私有作用域中形成闭包,并且可以访问或者修改私有的状态;

  function CoolModule() {

    var something = “cool”;

  var another = [1, 2, 3];

  function doSomething() {

  console.log(something);

  }

  function doAnother() {

    console.log(another.join(“!”));

  }

  return {

    doSomething: doSomething,

    doAnother: doAnother

  };

 }

  var foo = CoolModule();

  foo.doSomething();

你不知道的JavaScript(作用域和闭包)的更多相关文章

  1. javascript作用域和闭包之我见

    javascript作用域和闭包之我见 看了<你不知道的JavaScript(上卷)>的第一部分--作用域和闭包,感受颇深,遂写一篇读书笔记加深印象.路过的大牛欢迎指点,对这方面不懂的同学 ...

  2. JavaScript 作用域和闭包——另一个角度:扩展你对作用域和闭包的认识【翻译&plus;整理】

    原文地址 --这篇文章有点意思,可以扩展你对作用域和闭包的认识. 本文内容 背景 作用域 闭包 臭名昭著的循环问题 自调用函数(匿名函数) 其他 我认为,尝试向别人解释 JavaScript 作用域和 ...

  3. 你不知道的JavaScript --- 作用域相关

    本篇是<你不知道的JavaScript>的读书笔记 什么是作用域? 程序离不变量,那么变量存储在哪里?程序需要时如何找到他们? 这些问题说明需要一套设计良好的规则来存储变量, 并且之后可以 ...

  4. JavaScript作用域和闭包

    在JavaScript中,作用域是执行代码的上下文.作用域有3种类型: 1.全局作用域 2.局部作用域---(又叫函数作用域) 3.eval作用域 var foo =0;//全局作用域console. ...

  5. 举例详细说明javascript作用域、闭包原理以及性能问题(转)

    转自:http://www.cnblogs.com/mrsunny/archive/2011/11/03/2233978.html 这可能是每一个jser都曾经为之头疼的却又非常经典的问题,关系到内存 ...

  6. JavaScript 作用域和闭包

    作用域的嵌套将形成作用域链,函数的嵌套将形成闭包.闭包与作用域链是 JavaScript 区别于其它语言的重要特性之一. 作用域 JavaScript 中有两种作用域:函数作用域和全局作用域. 在一个 ...

  7. javascript作用域、闭包、对象与原型链

    原文作者总结得特别好,自己收藏一下.^-^ 1.作用域1.1函数作用域JS的在函数中定义的局部变量只对这个函数内部可见,称之谓函数作用域.它没有块级作用域(因此if.for等语句中的花括号不是独立作用 ...

  8. javascript作用域与闭包

    Javasript作用域概要 在javascript中,作用域是执行代码的上下文,作用域有三种类型: 1)  全局作用域 2)  局部作用域(函数作用域) 3)  eval作用域 var foo = ...

  9. JavaScript作用域与闭包总结

    1.全局作用域 所有浏览器都支持 window 对象,它表示浏览器窗口,JavaScript 全局对象.函数以及变量均自动成为 window 对象的成员.所以,全局变量是 window 对象的属性,全 ...

  10. javascript——作用域与闭包

    http://www.cnblogs.com/lucio-yr/p/4047972.html 一.作用域: 在函数内部:用 var 声明的表示局部变量,未用var的是全局变量. 作用域取决于变量定义时 ...

随机推荐

  1. JS&colon;window&period;onload的使用

    1.最简单的调用方式 直接写到html的body标签里面,如: (html) (body onload="func()") (/body) (/html) 2.在JS语句调用 (s ...

  2. 【Other】推荐点好听的钢琴曲

    2013-12-13 16:19 匿名 | 浏览 138977 次 音乐钢琴 推荐点好听的钢琴曲,纯音乐也可以thanks!!! 2013-12-14 19:34 网友采纳 热心网友 巴洛克:帕海贝尔 ...

  3. &period;Net连接数据库-曾&comma;删&comma;改&comma;查&lpar;AOD&period;Net&rpar;

    连接数据库 static void Main(string[] args) { //SqlConnection conn = new SqlConnection();//实例化 //conn.Conn ...

  4. JVM垃圾回收机制总结&lpar;5&rpar; :JDK垃圾收集器的配置命令

    以下配置主要针对分代垃圾回收算法而言. 堆大小设置 年轻代的设置很关键 JVM中最大堆大小有三方面限制:相关操作系统的数据模型(32-bt还是64-bit)限制:系统的可用虚拟内存限制:系统的可用物理 ...

  5. 美H1B签证额满,硕士以上学位仍可申请

    美国公民与移民服务局6月1日宣布,2007会计年度的工作签证H1B名额已经用完,下年度的申请从明年4月1日开始.但在美国获得硕士以上学位的高学历者仍可申请. 据北美世界日报报道,美国移民律师协会对连续 ...

  6. Linux c codeblock的使用(一):新建一个工程

    (1)点击New->Project,出现如下图所示,然后再选择Console application,点击Go (2)点击Next (3)根据自己的需求选择特定的语言(前提是你的系统上有这个语言 ...

  7. &lbrack;Micropython&rsqb;TPYBoard v202 v102&plus;v202 家庭无线温湿度检测

     一.实验器件 1.TPYBoard v102 1块 2.TPYBoard v202 1块 3.Nokia 5110LCD显示屏 1块 4.DHT11温湿度传感器 1个 5.micro USB 数据线 ...

  8. 实现简单的shell sed替换功能

    通过脚本传参数可以实现替换 # -*-coding:utf-8-*- # Author:sunhao import sys f = open('yesterday','r',encoding='utf ...

  9. python json格式字符串转换为字典格式

    不废话,看代码 #_*_ coding:utf- _*_ import os import json course=open('C:\\Users\\ly199\\Desktop\\list.txt' ...

  10. sqlserver 时间字符串转化为时间格式

    ),),),),,)) select substring('D:\\files,3,len('D:\\files)-2) --去掉前两位路径D: