js手写call、bind、apply

目录

callapplybind有两种实现方式,第一种是隐式绑定,第二种是通过new

无论是通过隐式绑定实现还是通过new实现,核心都是针对this的绑定规则

具体关于this的绑定规则可以看我这一篇博客
this绑定规则

call与apply

我们先回想一下call的使用方法

javascript 复制代码
function foo() {
    return this
}
console.log(foo.call({ name: 18 }))

接下来让我们一步步实现

首先call是一个实例方法,所以我们的myCall也需要设置在Function.prototype

javascript 复制代码
Function.prototype.myCall = function () {
}

call会传入多个参数,第一个是被绑定的this,剩下的是函数调用的参数列表

javascript 复制代码
Function.prototype.myCall = function (that, ...args) {
}

接下来我们就需要在myCall中调用函数,我们可以通过this()的形式直接调用函数,因为foo.myCall本身就是一种隐式绑定

因为函数可能会有返回结果,所以我们需要将返回结果给返回出去

javascript 复制代码
Function.prototype.myCall = function (that, ...args) {
    const value = this(...args)
    return value
}

接下来我们需要将thisthis绑定到我们指定的that身上,我们可以在that上新增一个属性,用这个属性来存放函数

然而我们并不能确定that身上拥有哪些属性,没有哪些属性,如果我们的函数存放到了原有的属性上则可能会发生错误,这表明了我们无法确定存放函数的属性名

幸运的是ES6中推出了新的基本类型Symbol,使用它我们可以很轻松的得到唯一的属性名

具体关于Symbol可以看我这篇文章
Symbol

javascript 复制代码
Function.prototype.myCall = function (that, ...args) {
    const fn = Symbol("fn")
    that[fn] = this
    const value = that[fn](...args)
    return value
}

我们并不希望用户能在函数中访问到我们的fn,在fn运行完之后我们也需要将fn删除,所以我们需要使用属性描述符来配置fn

具体属性描述符可以看我这篇文章
属性描述符

javascript 复制代码
Function.prototype.myCall = function (that, ...args) {
    const fn = Symbol("fn")
    Object.defineProperty(that, fn, {
        value: this,
        enumerable: false,
        configurable: true
    })
    const value = that[fn](...args)
    delete that[fn]
    return value
}

最后我们还需要针对that来判断一下,如果传入的thatnull或者undefined,我们需要将函数绑定到window,如果传入的不是一个对象,我们就需要将它装箱

javascript 复制代码
Function.prototype.myCall = function (that, ...args) {
    that = (typeof that === "null" || typeof that === "undefined") ? window : Object(that)
    const fn = Symbol("fn")
    Object.defineProperty(that, fn, {
        value: this,
        enumerable: false,
        configurable: true
    })
    const value = that[fn](...args)
    delete that[fn]
    return value
}

最后我们来测试一下

javascript 复制代码
function foo() {
    return this
}
console.log(foo.myCall({ name: 18 }))

apply

applycall类似,只有在传入参数上不一样,call是将函数参数一个一个传入,而apply是将参数合为一个数组传入

javascript 复制代码
Function.prototype.myApply = function (that, args) {
    that = (typeof that === "null" || typeof that === "undefined") ? window : Object(that)
    const fn = Symbol("fn")
    Object.defineProperty(that, fn, {
        value: this,
        enumerable: false,
        configurable: true
    })
    const value = that[fn](...args)
    delete that[fn]
    return value
}

bind

bind和以上两个很类似,区别在于bind返回一个函数,这个函数不能再通过callapply来更改this指向

javascript 复制代码
Function.prototype.myBind = function (that, ...args) {
    that = (typeof that === "null" || typeof that === "undefined") ? window : Object(that)
    const fn = Symbol("fn")
    return (...args2) => {
        const allArgs = args.concat(args2)
        Object.defineProperty(that, fn, {
            value: this,
            enumerable: false,
            configurable: false
        })
        const value = that[fn](...allArgs)
        return value
    }
}

需要注意的是,我们返回的函数不能删除fn,如果删除了fn的话运行会报错,因为找不到这个属性

如果在返回的函数上进行call或者apply也不会更改新函数的this,因为无论是call还是apply都是在原先函数上套了一个新函数,由新函数隐式调用原函数,但经过bind绑定过后的函数被一个新函数包裹,经历了两次调用,所以原函数的this指向不会丢失

测试结果

javascript 复制代码
function foo() {
    return this
}
const newFoo = foo.myBind({ name: 18 })
console.log(newFoo())
console.log(newFoo.myCall(window))
console.log(newFoo.apply(window))
相关推荐
Highcharts.js8 分钟前
缺失数据可视化图表开发实战|Highcharts创建人员出生统计面积图表示例
开发语言·前端·javascript·信息可视化·highcharts·图表开发
测试员周周5 小时前
【Appium 系列】第16节-WebView-H5上下文切换 — 混合应用的自动化难点
运维·开发语言·人工智能·功能测试·appium·自动化·测试用例
杜子不疼.7 小时前
【C++ AI 大模型接入 SDK】 - DeepSeek 模型接入(上)
开发语言·c++·chatgpt
加号37 小时前
【C#】 串口通信技术深度解析及实现
开发语言·c#
sycmancia8 小时前
Qt——编辑交互功能的实现
开发语言·qt
石山代码8 小时前
C++ 内存分区 堆区
java·开发语言·c++
无风听海9 小时前
C# 隐式转换深度解析
java·开发语言·c#
放下华子我只抽RuiKe59 小时前
React 从入门到生产(四):自定义 Hook
前端·javascript·人工智能·深度学习·react.js·自然语言处理·前端框架
一只大袋鼠9 小时前
Git 进阶(二):分支管理、暂存栈、远程仓库与多人协作
java·开发语言·git
LuminousCPP10 小时前
数据结构 - 线性表第四篇:C 语言通讯录优化升级全记录(踩坑 + 思考)
c语言·开发语言·数据结构·经验分享·笔记·学习