1.作用域和作用域链

1.作用域

一套变量查找的规则, 运行时代码中的某些特定部分中的变量, 函数和对象的可访问性.
作用域就是一个独立的地盘, 让变量不会外泄.
作用域最大的用处就是隔离变量, 不同作用域下的同名变量不会有冲突

1.1 全局作用域 与 函数作用域

代码中任何地方都可以访问到的对象, 拥有全局作用域,

拥有全局作用域的几种情况

  1. 最外层函数, 以及在最外层函数定义的变量
  2. 所有未定义直接赋值的变量自动声明为全局
  3. 所有window对象的属性

函数作用域 声明在函数内部的变量函数对象

1.2 作用域分层的

内层的作用域可以访问到外层的, 反之不行

1.3 注意

块语句 {}, if switch for while 不会创建新的作用域

1.4 [[scope]]]

[[scope]]: 当前执行的上下文,

  1. 函数创建时,生成的一个JS内部的隐式创建
  2. 函数存储作用域链的容器,作用域链
  3. 如果一个变量(en-US)或者其他表达式不”在当前的作用域中”, 那么它就是不可用的
  4. 作用域也可以根据代码层次分层,以便子作用域可以访问父作用域,通常是指沿着链式的作用域链查找,而不能从父作用域引用子作用域中变量和引用

2.作用域链

scope chain: 决定了各级上下文的代码在访问变量和函数的顺序
作用域链: AO/GO

  1. AO: 函数执行期的上下文
  2. GO: 全局执行期的上下文
  3. 函数执行之前会产生AO
  4. 函数执行完成以后,AO是要销毁的
  5. AO是一个即时的存储容器

1.什么是自由变量

首先认识一下什么叫做 自由变量 。如下代码中,console.log(a)要得到a变量,但是在当前的作用域中没有定义a(可对比一下b)。当前作用域没有定义的变量,这成为 自由变量 。自由变量的值如何得到 —— 向父级作用域寻找(注意:这种说法并不严谨,下文会重点解释)。

  1. var a = 100
  2. function fn() {
  3. var b = 200
  4. console.log(a) // 这里的a在这里就是一个自由变量
  5. console.log(b)
  6. }
  7. fn()

2.什么是作用域链

如果父级也没呢?再一层一层向上寻找,直到找到全局作用域还是没找到,就宣布放弃。这种一层一层的关系,就是 作用域链 。

  1. var a = 100
  2. function F1() {
  3. var b = 200
  4. function F2() {
  5. var c = 300
  6. console.log(a) // 自由变量,顺作用域链向父作用域找
  7. console.log(b) // 自由变量,顺作用域链向父作用域找
  8. console.log(c) // 本作用域的变量
  9. }
  10. F2()
  11. }
  12. F1()

3.关于自由变量的取值

关于自由变量的值,上文提到要到父作用域中取,其实有时候这种解释会产生歧义。

  1. var x = 10;
  2. function fn() {
  3. console.log(x)
  4. }
  5. function show(f) {
  6. var x = 20;
  7. (function () {
  8. fn();
  9. })()
  10. }
  11. show(fn);

在fn函数中,取自由变量x的值时,要到哪个作用域中取?——要到创建fn函数的那个作用域中取,无论fn函数将在哪里调用
所以,不要在用以上说法了。相比而言,用这句话描述会更加贴切:要到创建这个函数的那个域”。
作用域中取值,这里强调的是“创建”,而不是“调用”
,切记切记——其实这就是所谓的”静态作用域“或”词法作用域

  1. var a = 10
  2. function fn() {
  3. var b = 20
  4. function bar() {
  5. console.log(a + b) //30
  6. }
  7. return bar
  8. }
  9. var x = fn(),
  10. b = 200
  11. x() //bar()


fn()返回的是bar函数,赋值给x。执行x(),即执行bar函数代码。取b的值时,直接在fn作用域取出。取a的值时,试图在fn作用域取,但是取不到,只能转向创建fn的那个作用域中去查找,结果找到了,所以最后的结果是30

  1. var obj = {
  2. name: '蓝轨迹',
  3. address: '北京',
  4. teach: function() {}
  5. }
  6. function test(a, b) {
  7. }
  8. // 函数也是一种对象类型 引用类型 引用值
  9. // test.name test.length test.prototype
  10. // 对象 => 有些属性我们无法访问的
  11. // JS引擎内部固有的隐式属性
  1. // 案例1
  2. function a(){
  3. function b(){
  4. var b = 2;
  5. }
  6. var a = 1;
  7. b();
  8. }
  9. var c = 3;
  10. a();
  1. test2();
  2. function tset2() {}
  3. test();
  4. var test = function(){}
  5. // GO = {
  6. // test: undefined => f (){}
  7. // test2: f test() {}
  8. // }
  1. function a(){
  2. function b(){
  3. function c(){
  4. }
  5. c();
  6. }
  7. b();
  8. }
  9. a();
  10. // a定义: a.[[scope]] => 0 : GO
  11. // a执行: a.[[scope]] => 0 : a => AO
  12. // 1 : GO
  13. // b定义: b.[[scope]] => 0 : a => AO
  14. // 1 : GO
  15. // b执行: b.[[scope]] => 0 : b => AO
  16. // 1 : a => AO
  17. // 2 : GO
  18. // c定义: c.[[scope]] => 0 : b => AO
  19. // 1 : a => AO
  20. // 2 : GO
  21. // c执行: c.[[scope]] => 0 : c => AO
  22. // 1 : b => AO
  23. // 2 : a => AO
  24. // 3 : GO
  25. // c执行完: c.[[scope]] => 0 : b => AO
  26. // 1 : a => AO
  27. // 2 : GO
  28. // b执行完: b.[[scope]] => 0 : a => AO
  29. // 1 : GO
  30. // c.[[scope]] 不存在
  31. // a执行完: a.[[scope]] => 0: GO
  32. // b.[[scope]]不存在

2.闭包

  1. 当内部函数被返回到外部的时候,一定会产生闭包
  2. 闭包会产生原来的作用域链不释放
  3. 过渡闭包可能会导致内存泄露或加载过慢

一个函数和对其周围状态(lexical enviroment,词法环境)的引用捆绑在一起(或者说函数被引用包围),这样的组合就是闭包(closure).
也就是说,闭包让你可以在一个内层函数中访问到其外层函数的作用域.
在 JavaScript中,每当创建一个函数,闭包就会在函数创建的同时被创建出来.

  1. function test1() {
  2. function test2() {
  3. var b = 2;
  4. console.log(a);
  5. }
  6. var a = 1;
  7. return test2();
  8. }
  9. var c = 3;
  10. var test3 = test1();
  1. function test() {
  2. var n = 100;
  3. function add() {
  4. n++;
  5. console.log(n)
  6. }
  7. function reduce() {
  8. n--;
  9. console.log(n);
  10. }
  11. return [add, reduce];
  12. }
  13. var arr = test();
  14. arr[0](); // 101
  15. arr[1](); // 100
  1. function breadMgr(num) {
  2. var breadNum = arguments[0] || 10;
  3. function supply() {
  4. breadNum += 10;
  5. console.log(breadNum);
  6. }
  7. function sale() {
  8. breadNum--;
  9. console.log(breadNum);
  10. }
  11. return [supply, sale];
  12. }
  13. var breadMgr = breadMgr(50);
  14. breadMgr[0](); // 60
  1. function sunSched() {
  2. var sunSched = '';
  3. var operation = {
  4. setSched:function (thing) {
  5. sunSched = thing;
  6. },
  7. showSched:function() {
  8. console.log('My schedule on sunday is ' + sunSched);
  9. }
  10. }
  11. return operation;
  12. }
  13. var sunSched = sunSched();
  14. sunSched.setSched('studying');
  15. sunSched.showSched(); // My schedule on sunday is studying