手撕call-apply-bind内置函数的实现原理

这篇博客探讨了JavaScript中改变`this`指向的三种常见方法:call、apply和bind。通过示例展示了它们的使用场景和区别,包括call和apply参数传递的不同,以及bind返回新函数的特性。此外,还提供了这三种方法的自定义实现,加深了对它们工作原理的理解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

有关这三个函数,在实际工作上,应该是用的比较多的,改变this指向的三种方式。call和apply类似,唯一的区别就是参数的传递方式有所不同而已,而bind函数是对需要执行的函数进行this绑定后返回一个新函数,且新函数的this不可再次改变。

call: call() 方法使用一个指定的 this 值和单独给出的一个或多个参数来调用一个函数。

该方法的语法和作用与 apply() 方法类似,只有一个区别,就是 call() 方法接受的是一个参数列表,而 apply() 方法接受的是一个包含多个参数的数组。

当前执行结果都是在非严格模式下。

bind() 方法创建一个新的函数,在 bind() 被调用时,这个新函数的 this 被指定为 bind() 的第一个参数,而其余参数将作为新函数的参数,供调用时使用。

const foo = function (a, b) {
      console.log(a + b, this);
    }
    Function.prototype.myCall = function (thisArg, ...params) {
      // null || undefined   => if(thisArg == null)
      if (thisArg === null || typeof thisArg === "undefined") return this(params);
      // 原始类型的值 转为包装类
      if (!/^(object|function)$/i.test(typeof thisArg)) thisArg = new Object(thisArg)
      const sb = Symbol('fn');
      // thisArg[sb] = this;
      Object.defineProperty(thisArg, sb, {
        // enumerable: false,
        value: this,
        configurable: true,
        // writable: false
      })
      const res = thisArg[sb](...params);
      // 此时删除了 但是在执行函数的过程时:控制台可以看见this中有该属性,
      // 因为执行时 该属性还不能删除掉
      delete thisArg[sb];
      return res;
    }
    Function.prototype.myApply = function (thisArg, params) {
      return this.myCall(thisArg, ...params);
    }
    Function.prototype.myBind = function (thisArg, ...params) {
      // 使用箭头函数 this不可再次改变
      return (...args) => {
        this.myCall(thisArg, ...params, ...args);
      }
    }
    foo.myCall({ name: '毛毛' }, 1, 2);
    foo.myCall("2", 1, 2);
    foo.myApply(20, [1, 4]);
    foo.myBind({ name: "Mao" }, 1)(2);
    foo.myBind({ name: "Mao" }, 1).bind(null)(12);
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

尤雨东

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值