一道经典的 JavaScript 面试题

已知以下一段代码,请回答问题。

function Foo (){
    getName = function (){
	alert(1);
    };
    return this;
}
 
Foo.getName = function (){
    alert(2);
};
 
Foo.prototype.getName = function () {
    alert(3);
};
 
var getName = function (){
    alert(4);
};
 
function getName (){
    alert(5);
};
 
Foo.getName();
getName();
Foo().getName();
getName();
new Foo.getName();
new Foo().getName();
new new Foo().getName();

请回答:各部分的输出是什么?
2
4
1
1
2
3
3

解答思路:

  1. 最开始的变量对象:
VO (globalContext) = {
	Foo:{
		<reference to function>,
		return this
	}
	getName: <reference to function>{alert(5)}
}
  1. 执行 Foo.getName = function (){alert(2);}
VO (globalContext) = {
	Foo:{
		<reference to function>,
		getName: <reference to function>{alert(2)},
		return this
	}	
	getName: <reference to function>{alert(5)}
  1. 执行 Foo.prototype.getName = function () {alert(3);};
VO (globalContext) = {
	Foo:{
		<reference to function>,
		getName: <reference to function>{alert(2)},
		prototype: {
			getName: <reference to function>{alert(3)}
		},
		return this
	}
	getName: <reference to function>{alert(5)}	
}
  1. 执行 var getName = function (){alert(4);};
VO (globalContext) = {
	Foo:{
		<reference to function>,
		getName: <reference to function>{alert(2)},
		prototype: {
			getName: <reference to function>{alert(3)}
		},
		return this
	}
	getName: <reference to function>{alert(4)}	
}
  1. 执行 Foo.getName():
    从 VO 中看到 :alert(2)

  2. 执行 getName():
    从 VO 中看到 :alert(4)

  3. 执行 Foo().getName():
    Foo() 调用之后返回 this(指向window),所以 Foo().getName() 等价于 window.getName()
    但是Foo() 调用改变了 VO

VO (globalContext) = {
	Foo:{
		<reference to function>,
		getName: <reference to function>{alert(2)},
		prototype: {
			getName: <reference to function>{alert(3)}
		},
		return this
	}
	getName: <reference to function>{alert(1)}	
}

所以 Foo().getName() 最后返回 alert(1)

  1. 执行 new Foo.getName():
    因为 new 操作符的优先级低于属性访问符,所以先执行 Foo.getName(),返回 alert(2)

  2. 执行 new Foo().getName():
    Foo() 和 new Foo() 相比, new Foo() 优先级更高,所以先执行 new Foo(),new 出来的对象继承了 Foo.prototype ,因此 alert(3)

  3. 执行 new new Foo().getName():
    先执行 new Foo(),所以alert(3)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章