js经典面试题01

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

//请写出以下输出结果:
Foo.getName();
getName();
Foo().getName();
getName();
new Foo.getName();
new Foo().getName();
new new Foo().getName();

由于经常有人拿这个面试提问所以整理了一下保存起来

解析:

1、Foo.getName(); //2

Foo是一个函数,也可以说是一个对象,所以它也可以挂载一些属性和方法。
所以结果执行的是Foo对象的一个叫做getName()的属性,而1、4、5中的getName都是作为函数存在,所以可以排除1、4、5

剩下两个中,2是Foo对象自身的属性,3是Foo对象原型链上的属性,而自身属性的优先级高于原型链上的属性,所以执行结果是2


2、getName(); //4

结果执行的是getName函数,而题目代码中有3个相关函数,分别是1、4、5

1中的getName是定义在Foo函数中的函数,由于Foo尚未执行,因此它没有暴露出来,无法被外部调用,可以排除

4和5都可以被正常调用,关键在调用先后问题

由于5是普通函数(优先级最高),4是匿名函数;js解析时会将5提前至最上方优先解析,而后面解析的4会将5覆盖,所以执行结果是4


3、Foo().getName(); //1

结果执行的是Foo函数,Foo函数中有个返回值是this;this被普通函数调用后,指向的对象一定是window对象,所以此处的结果已经可以解析为window.getName(),即调用getName()函数,所以最后执行的就是window.getName,所以输出1;


4、getName(); //1

在上面已经更改全局的getName,执行getName即是执行window.getName;所以依然是1


5、new Foo.getName(); //2

new 操作符在实例化构造器的时候,会执行构造器函数,也就是说,foo.getName会执行,输出2


6、new Foo().getName(); //3

new操作符的优先级较高,所以会先new foo()得到一个实例,然后再执行实例的getName方法,这个时候,实例的构造器里没有getName方法,就会执行构造器原型上的getName方法


7、new new Foo().getName(); //3

先执行new foo()得到一个实例,然后在new 这个实例的getName方法,这个时候会执行这个方法,所以输出3


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,768评论 19 139
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 13,831评论 1 32
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 32,080评论 18 399
  • 【壹】 最近一直有些凌乱的生活节奏,突然在这个午后睡不着的时间里,想静下来想一想了。不久以前有个好习惯,写一些类似...
    mellisa_901阅读 2,985评论 1 3
  • 因为有了人海,所以相遇便成了意外。初秋九月,天空依稀晴朗,大地绿意依然,秋风不时地吹起,透着习习凉意……景色依旧...
    易安言阅读 1,339评论 0 2