1.作用域和作用域链
1.作用域
一套变量查找的规则, 运行时代码中的某些特定部分中的变量, 函数和对象的可访问性.
作用域就是一个独立的地盘, 让变量不会外泄.
作用域最大的用处就是隔离变量, 不同作用域下的同名变量不会有冲突
1.1 全局作用域 与 函数作用域
代码中任何地方都可以访问到的对象, 拥有全局作用域,
拥有全局作用域的几种情况
- 最外层函数, 以及在最外层函数定义的变量
- 所有未定义直接赋值的变量自动声明为全局
- 所有window对象的属性
函数作用域 声明在函数内部的变量函数对象
1.2 作用域分层的
内层的作用域可以访问到外层的, 反之不行
1.3 注意
块语句 {}, if switch for while 不会创建新的作用域
1.4 [[scope]]]
[[scope]]: 当前执行的上下文,
- 函数创建时,生成的一个JS内部的隐式创建
- 函数存储作用域链的容器,作用域链
- 如果一个变量(en-US)或者其他表达式不”在当前的作用域中”, 那么它就是不可用的
- 作用域也可以根据代码层次分层,以便子作用域可以访问父作用域,通常是指沿着链式的作用域链查找,而不能从父作用域引用子作用域中变量和引用
2.作用域链
scope chain: 决定了各级上下文的代码在访问变量和函数的顺序
作用域链: AO/GO
- AO: 函数执行期的上下文
- GO: 全局执行期的上下文
- 函数执行之前会产生AO
- 函数执行完成以后,AO是要销毁的
- AO是一个即时的存储容器
1.什么是自由变量
首先认识一下什么叫做 自由变量 。如下代码中,console.log(a)要得到a变量,但是在当前的作用域中没有定义a(可对比一下b)。当前作用域没有定义的变量,这成为 自由变量 。自由变量的值如何得到 —— 向父级作用域寻找(注意:这种说法并不严谨,下文会重点解释)。
var a = 100
function fn() {
var b = 200
console.log(a) // 这里的a在这里就是一个自由变量
console.log(b)
}
fn()
2.什么是作用域链
如果父级也没呢?再一层一层向上寻找,直到找到全局作用域还是没找到,就宣布放弃。这种一层一层的关系,就是 作用域链 。
var a = 100
function F1() {
var b = 200
function F2() {
var c = 300
console.log(a) // 自由变量,顺作用域链向父作用域找
console.log(b) // 自由变量,顺作用域链向父作用域找
console.log(c) // 本作用域的变量
}
F2()
}
F1()
3.关于自由变量的取值
关于自由变量的值,上文提到要到父作用域中取,其实有时候这种解释会产生歧义。
var x = 10;
function fn() {
console.log(x)
}
function show(f) {
var x = 20;
(function () {
fn();
})()
}
show(fn);
在fn函数中,取自由变量x的值时,要到哪个作用域中取?——要到创建fn函数的那个作用域中取,无论fn函数将在哪里调用。
所以,不要在用以上说法了。相比而言,用这句话描述会更加贴切:要到创建这个函数的那个域”。
作用域中取值,这里强调的是“创建”,而不是“调用”,切记切记——其实这就是所谓的”静态作用域“或”词法作用域“
var a = 10
function fn() {
var b = 20
function bar() {
console.log(a + b) //30
}
return bar
}
var x = fn(),
b = 200
x() //bar()
fn()返回的是bar函数,赋值给x。执行x(),即执行bar函数代码。取b的值时,直接在fn作用域取出。取a的值时,试图在fn作用域取,但是取不到,只能转向创建fn的那个作用域中去查找,结果找到了,所以最后的结果是30
var obj = {
name: '蓝轨迹',
address: '北京',
teach: function() {}
}
function test(a, b) {
}
// 函数也是一种对象类型 引用类型 引用值
// test.name test.length test.prototype
// 对象 => 有些属性我们无法访问的
// JS引擎内部固有的隐式属性
// 案例1
function a(){
function b(){
var b = 2;
}
var a = 1;
b();
}
var c = 3;
a();
test2();
function tset2() {}
test();
var test = function(){}
// GO = {
// test: undefined => f (){}
// test2: f test() {}
// }
function a(){
function b(){
function c(){
}
c();
}
b();
}
a();
// a定义: a.[[scope]] => 0 : GO
// a执行: a.[[scope]] => 0 : a => AO
// 1 : GO
// b定义: b.[[scope]] => 0 : a => AO
// 1 : GO
// b执行: b.[[scope]] => 0 : b => AO
// 1 : a => AO
// 2 : GO
// c定义: c.[[scope]] => 0 : b => AO
// 1 : a => AO
// 2 : GO
// c执行: c.[[scope]] => 0 : c => AO
// 1 : b => AO
// 2 : a => AO
// 3 : GO
// c执行完: c.[[scope]] => 0 : b => AO
// 1 : a => AO
// 2 : GO
// b执行完: b.[[scope]] => 0 : a => AO
// 1 : GO
// c.[[scope]] 不存在
// a执行完: a.[[scope]] => 0: GO
// b.[[scope]]不存在
2.闭包
- 当内部函数被返回到外部的时候,一定会产生闭包
- 闭包会产生原来的作用域链不释放
- 过渡闭包可能会导致内存泄露或加载过慢
一个函数和对其周围状态(lexical enviroment,词法环境)的引用捆绑在一起(或者说函数被引用包围),这样的组合就是闭包(closure).
也就是说,闭包让你可以在一个内层函数中访问到其外层函数的作用域.
在 JavaScript中,每当创建一个函数,闭包就会在函数创建的同时被创建出来.
function test1() {
function test2() {
var b = 2;
console.log(a);
}
var a = 1;
return test2();
}
var c = 3;
var test3 = test1();
function test() {
var n = 100;
function add() {
n++;
console.log(n)
}
function reduce() {
n--;
console.log(n);
}
return [add, reduce];
}
var arr = test();
arr[0](); // 101
arr[1](); // 100
function breadMgr(num) {
var breadNum = arguments[0] || 10;
function supply() {
breadNum += 10;
console.log(breadNum);
}
function sale() {
breadNum--;
console.log(breadNum);
}
return [supply, sale];
}
var breadMgr = breadMgr(50);
breadMgr[0](); // 60
function sunSched() {
var sunSched = '';
var operation = {
setSched:function (thing) {
sunSched = thing;
},
showSched:function() {
console.log('My schedule on sunday is ' + sunSched);
}
}
return operation;
}
var sunSched = sunSched();
sunSched.setSched('studying');
sunSched.showSched(); // My schedule on sunday is studying