一道经典的 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)

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值