在学习预编译之前,我们要先了解一下js的运行步骤:
(1)函数声明整体提升:不管你的函数声明写在哪里,系统总是会将其提升到逻辑最前面,所以不管你是在函数的上面或下面调用函数,其实都是在下面调用的。
(2)变量仅声明提升:如 var a = 123;则将其拆成两个部分 即 声明var a位置提升到逻辑最前端,赋值a=123不提升。所以若是在声明赋值变量前进行调用,则会显示undefined,而不会报错。
(3)imply global 暗示全局变量:即任何变量,如果变量未经声明就赋值,此变量就为全局对象所有。即:
function test(){
a = 10; //在局部里面赋值,没有进行var声明
}
因为没有进行var声明,相当于在全局域里定义,即window.a = 10;
var a = b = 123;//局部里
上面代码的执行顺序:先将123赋给b,其次是b的值赋给a,但是声明只声明了a,没有声明b,所以此时b归window所有。
(4)一切声明的全局变量,全是window的属性,即归window所有,而调用全局变量,其实就是调用window.变量。(window 就是全局的域)即:
var b = 234; //在全局域里面声明定义
即 相当于window.b = 234。
下面就通过check这个函数来解释预编译的四个步骤
function check(a) {
console.log(a);
var a = 123; //变量声明和定义
console.log(a);
function a() {} //函数声明
console.log(a);
var b = function () {} //变量声明和定义
console.log(b);
function d() {} //函数声明
}
check(1);
AO {}
从上面的代码中,我们可以知道形参是a,也定义了变量b。
AO {
a : undefined,
b : undefined,
}
函数执行传入的实参是1,所以形参a=1。
AO {
a : 1,
b : undefined,
}
a的值变成了函数体(存在同名的情况,直接覆盖以前的值),还添加了一个d属性,值为函数体。
AO {
a : function a() {},
b : undefined,
d : function d() {}
}
以上预编译过程就结束了,接下来就是js三步曲中的第三步------解释执行。(解释执行的时候,AO对象里的属性值也会发生改变,而每次想要输出的变量,其实就是在AO对象中找对应的值)
//开始js的解释执行,修改AO对象
function check(a) {
// 此时的AO对象就是预编译结束之后的样子
// AO {
// a : function a() {},
// b : undefined,
// d : function d() {}
// }
console.log(a); //输出function a() {}
var a = 123; //预编译已经将var a提升执行过了,所以直接执行赋值a=123
// AO {
// a : 123,
// b : undefined,
// d : function d() {}
// }
console.log(a); //输出123
function a() {} //预编译已经提升声明了,被优先执行了,所以不看了
console.log(a); //输出123
var b = function () {} //预编译已经将变量b的声明提升执行了,所以只用看赋值
// AO {
// a : 123,
// b : function() {},
// d : function d() {}
// }
console.log(b); //输出function() {}
function d() {} //预编译已经提升声明了,被优先执行了,所以不看了
}
check(1); //函数执行,实参为1
现在就通过下面的代码来练习一下全局和局部一起的预编译和它们的执行过程:
console.log(test);
function test(test) {
console.log(test);
var test = 234;
console.log(test);
function test() { }
}
test(1);
var test = 123;
(1) 创建GO对象:全局预编译的情况下,会生成Global Object对象,其实和AO同理,只是换了个名字。(GO === window)
GO {}
(2)找变量声明作为GO属性名,值为undefined
GO {
test : undefined
}
(3)找函数声明,值赋予函数体
GO {
test : function test(test) {...}
}
(此时的AO对象为函数test里的)
(1)创建AO对象
AO {}
(2)找形参和变量声明赋值undefined
AO {
test : undefined
}
(3)实参形参统一
AO {
test: 1
}
(4) 找函数声明,值赋予函数体
AO {
test: function test() {}
}
// js开始解释执行
//全局
// GO {
// test : function test(test) {...}
// }
console.log(test); // function test(test) {...}
function test(test) {
// 局部
// AO {
// test: function test() {}
// }
console.log(test); // function test(){}
var test = 234;
// AO {
// test: 234
// }
console.log(test); // 234
function test() {}//预编译已经提升了声明,被优先执行了,所以不看了
}
test(1);
var test = 123;
// GO {
// test : 123
// }
console.log(test); //123
// js执行结束
注意:如果GO中和AO中都有test函数,若在局部中就先找AO的,但如果AO没有的话再找GO;若在全局中就找GO的用。(就近找,自己没有找爸爸,爸爸没有找爷爷等)
个人笔记,欢迎大家交流探讨!