zoukankan      html  css  js  c++  java
  • Function.prototype.bind、call与apply

    学习Function.prototype.bind、call与apply时,看到一篇博客,学到一些内容,但由于博客时间太久,根据官方文档对内容进行一些修正;下文为修正过内容

    前言

    前段时间面试遇见一题,题目内容大概是

        functionParent() {this.prop = 'parent';
        }
        
        Parent.prototype.get = function() {
            alert(this.prop);
        };
        
        Parent.prototype.show = function() {
            setTimeout(this.get, 100);
        };
        
        var child = new Parent();
        child.show(); // ?
    

    分析

    上述题目考察的是this的指向性这个经典问题。

    众所周知,setTimeout是window对象的一个属性,主要起到延迟给定函数执行的作用。setTimeout(fn, delay),因此this.get的这个this指向的是window对象,但是我们并没用在window对象上定义相应的get函数,所以会报错,而不是调用构造函数Parent的原型中的get方法。

    怎么改?

    经过上面的分析,我们知道setTimeout(this.get, 100)会报错,但是如果我们想正常调用并且alert出正确的值,应该怎么改?

    改动的原理很明确,就是将setTimeout中的this指向Parent.prototype,但是单纯的将this.get改成Parent.prototype.get仍然没有alert出真正的parent,而是undefined,原理仍然是原型中的get方法在通过实例对象child调用时内部的this仍然指向window对象。所以我们只能换种思路解决了

    思路1

        Parent.prototype.show = function(){var that = this;
            setTimeout(this.get.call(that), 100);
        }
    

    这种方法在很多框架中经常使用,即先存储this默认指向的作用域,然后改变函数内部绑定的作用域来实现。ps: 不通过that存储,直接传递this也可

    思路2

    ES5中对函数方面唯一扩展是新增了一个bind函数,用于劫持函数作用域,并预先添加更多参数,然后返回绑定了新作用域的函数。而现在我们也可以使用它,其实这个思路在MDN上有相应的介绍使用,哎,只怪自己的年少无知。。。

        Parent.prototype.show = function() {
            setTimeout(this.get.bind(this), 100);
        };
    

    bind/call/apply

    三者都是用于绑定函数作用域,区别如下:

    • call 是obj.method()到method(obj)的变换,返回函数调用结果,所需参数依次用逗号分割添加至obj尾部。
    • apply 功能同call,区别是传递参数的方式不是call的那种参数列表形式,而是以数组或类数组形式传递。
    • bind 返回绑定作用域后的一个新函数,不会执行函数

    我们可以通过apply方法实现bind函数:

    if (!Function.prototype.bind) {
      Function.prototype.bind = function(oThis) {
        if (typeof this !== 'function') {
          // closest thing possible to the ECMAScript 5
          // internal IsCallable function
          throw new TypeError('Function.prototype.bind - what is trying to be bound is not callable');
        }
    
        var aArgs   = Array.prototype.slice.call(arguments, 1),
            fToBind = this,
            fNOP    = function() {},
            fBound  = function() {
              // this instanceof fNOP === true时,说明返回的fBound被当做new的构造函数调用
              return fToBind.apply(this instanceof fNOP
                     ? this
                     : oThis,
                     // 获取调用时(fBound)的传参.bind 返回的函数入参往往是这么传递的
                     aArgs.concat(Array.prototype.slice.call(arguments)));
            };
    
        // 维护原型关系
        if (this.prototype) {
          // Function.prototype doesn't have a prototype property
          fNOP.prototype = this.prototype; 
        }
        // 下行的代码使fBound.prototype是fNOP的实例,因此
        // 返回的fBound若作为new的构造函数,new生成的新对象作为this传入fBound,新对象的__proto__就是fNOP的实例
        fBound.prototype = new fNOP();
    
        return fBound;
      };
    }
    

    另外,我们可以利用bind修复IE事件绑定attachEvent回调中的this问题,它总是指向window对象,而标准浏览器的addEventListener中的this则为其调用对象。

        function(el, type, fn) {
            el.attachEvent('on' + type, fn.bind(el, event));
        }
    
  • 相关阅读:
    JAVA设计模式之 訪问者模式【Visitor Pattern】
    xcode_6_beta.dmg
    windows linux 下安装mysql 报1045 等错误
    c语言基础学习10_文件操作01
    粘贴到vi/vim的代码,怎样避免向右不断缩进,保持原来的格式?解决vi/vim在粘贴中会在行首多很多缩进和空格的问题。
    在c语言中,数组 a[i++] 和数组 a[++i] 有区别吗? && 在c语言中,数组 a[0]++; 又是什么意思?
    如何把云端服务器上的file04.c文件传输到本地pc机windows系统上去呢?
    解决在SecurecCRT登录后,发现方向键、backspace(退格键)、delete(删除键)为乱码的问题
    自己定义iOS上双击Home键图切换
    MySQL联合多表更新和删除
  • 原文地址:https://www.cnblogs.com/ysk123/p/10000977.html
Copyright © 2011-2022 走看看