很多教程里都这样说:
函数声明的提升优先于变量提升:
(function(){
console.log(typeof a); //"function"
var a = 1;
function a(){};
})();
被解析为:
(function(){
function a(){};
var a = 1;
console.log(typeof a);
})();
然而我觉得是变量提升优先级大于函数声明,我觉得function a(){} 被提升到顶端会被解析为:
var a; a = function(){}; //这也可以解释为什么声明函数不能被外部调用。
这段代码应该被解析为:
(function(){
var a;
var a;
a = function(){};
console.log(typeof a);
})();
当然提升到顶端并不是真的提升到顶端,js是先解析在执行的,所谓提升到顶端其实就是在解析完后就已经声明好函数与变量了,所以就算你把断点打在你的函数第一句上,这些函数与变量也已经早就声明好了。
这就我的观点,这种观点是否正确呢?
就不给链接了,看的书太杂,好多知识点找不到出处。我无从论证他的正确性,如何看待,全看个人。
变量的问题,莫过于声明和赋值两个步骤,而这两个步骤是分开的。
函数声明被提升时,声明和赋值两个步骤都会被提升,而普通变量却只能提升声明步骤,而不能提升赋值步骤。
变量被提升过后,先对提升上来的所有对象统一执行一遍声明步骤,然后再对变量执行一次赋值步骤。而执行赋值步骤时,会优先执行函数变量的赋值步骤,再执行普通变量的赋值步骤。
当你明白这三点后,一切都豁然开朗了。
首先来看一个DEMO:
先提升两个a,然后执行函数的赋值步骤,a没有被赋值,故结果为function
再看一个:
先提升两个a,再执行函数的赋值步骤,因为在alert语句执行以前,还未执行a = 1的赋值步骤,函数不会被覆盖,故为function
来个最有说服力的:
在alert语句执行之前,a = 1步骤和函数赋值步骤均已执行,而且函数还在a = 1赋值语句之后,但是仍然输出number,就是因为函数的赋值步骤会先于a = 1的赋值步骤,函数被覆盖,故输出number。
一切关乎变量提升的代码,用这三点没有解释不清楚的。而至于你说的“为什么声明函数不能被外部调用”,这是作用域的问题,跟变量提升,其实并无直接关系。
以上。
欢迎拍砖...