博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
记一道经典前端题
阅读量:6651 次
发布时间:2019-06-25

本文共 2191 字,大约阅读时间需要 7 分钟。

题目如下:

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. Foo.getName() 输出为 2, 访问的是函数 Foo 上的静态属性,输出为 2。

    现在,尝试在函数内定义 getName 函数和在 Foo 原型上绑定 getName 函数, 都无法成功执行 Foo.getName(), 而以字面量创建对象的方式创建对象后,则能正常的执行 getName() 函数。

通过创建对象运行 getName() 定义在函数内部无法正常执行,即使它是全局变量。

结论: 由于函数本身是对象,通过函数绑定属性和方法属于静态方法 ,可以直接调用。绑定在原型上的属性和方法要创建对象后才能调用,在构造函数对象内部定义的方法无法通过对象调用。

  1. getName(); 结果输出为 4,而不是输出 5。这是因为JS 存在变量声明提升(所有声明的变量或声明的函数都会被提升到当前函数的顶部)。

    故代码执行顺序为:

    var getName;function getName() { alert(5); }// ... 省略代码getName = function () { alert(4) }复制代码

    最终执行 getName 输出为 4

    延伸题目:

    console.log( Foo )function Foo() {    console.log(1);}var Foo = 1复制代码

    Foo 的输出结果为?

  2. Foo().getName(); 输出值为1, 先执行 Foo() 函数,定义全局变量 getName, 之后调用全局对象的 getName() 方法, 返回 1。

    注意, Foo() 函数返回的 this 指向的是全局对象 window,所以调用的是全局对象 getName()。 函数里的 getName 绑定的是全局对象,通过 Foo 调用会报错。

    Node 下执行这条语句会报错,因为 node 没有全局对象 window, 所以无法调用 getName

  3. getName() 调用全局函数, 因为执行 Foo(), 更新了 getName 的值,所以返回 1。

  4. new Foo.getName() 考察了运算符的优先级。 . 的优先级高于 new, 相当于执行 new (Foo.getName)(), 相当于执行 getName 的构造函数,返回 2

  5. new Foo().getName() 执行方式为 (new Foo()).getName() 先生成 Foo 对象, 再执行 getName() 函数。 在 new Foo() 返回的是新创建的空对象,由于对象这时还没绑定属性 getName, 所以这时调用的是原型上的 getName, 结果返回3

    注意: 构造函数 return this,在执行 new 的时候,返回的是新创建的对象。

    延伸题目:

    function A() {    this.a = 2;    function B() {        this.a = 1;    }    return B();}console.log(new A());复制代码

    a的值是? 如果 return new B(); a的值是?

  6. new new Foo().getName() 可以改写为 new ((new Foo()).getName)() 先初始化实例,然后将原型对象上的 getName() 作为构造函数执行,结果返回 3

最终代码可以优化为

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

转载地址:http://tjyto.baihongyu.com/

你可能感兴趣的文章
vim命令助记图
查看>>
Python 的 csv 模块
查看>>
NDK使用之HelloWorld
查看>>
DNS浅析-搭建一个简单的主从服务器
查看>>
Linux 64位下jdk1.6和Weblogic11安装
查看>>
基于ZooKeeper的分布式Session实现
查看>>
ELK 日志分析平台搭建
查看>>
Nginx yum源
查看>>
Linux分区类型EXT2、EXT3、EXT4详解
查看>>
MTR--linux网络诊断工具
查看>>
基于rgw的多媒体转换网关原型设计
查看>>
如何取消hibernate hbm2ddl.auto 自动创建外键
查看>>
懒加载数据,在取出数据是容易出的bug....
查看>>
c#通过oledb获取excel文件表结构信息
查看>>
linux 正则
查看>>
php安装memcached扩展
查看>>
outlook收邮件时间与电脑时间不同步
查看>>
2012年常见问题FAQ
查看>>
VMware技术支持合作伙伴日-北京-2013年8月1日-幻灯片下载
查看>>
wget命令详解(附具体实例)
查看>>