深入理解 JavaScript 中的 call 方法实现

什么是 call 方法

在 JavaScript 中,call() 是一个内置的函数方法,它允许你在特定的作用域中调用函数,即可以改变函数执行时的 this 指向。

基本语法:

javascript 复制代码
func.call(thisArg, arg1, arg2, ...)

原生 call 方法示例

javascript 复制代码
const person = {
    name:"ling_wu"
}

function getName(){
    console.log(this.name)
}

// 使用call改变this指向
getName.call(person); //输出:"ling_wu"

这个过程实际上的操作是person这个对象中创建了一个getName的函数,并且这个函数的this指向了person这个对象

javascript 复制代码
person = {
    name:"ling_wu",
    getName:function(){
        console.log(this.name);
    }
}

getName执行完成后再从person对象中删除getName这个属性

手动实现 call 方法

有了上面的初步分析我们来初步实现

第一步:基本实现

javascript 复制代码
Function.prototype.myCall = function(context){
    // getName.myCall(person)这个位置getName调用的myCall,所以this指向getName
    context.fn = this;
    const result = context.fn();
    delete context.fn;
    return result;
}

第二步:考虑参数

call是可以传递参数的,并且是都喊分隔,可以使用es6的...args来接收

javascript 复制代码
Function.prototype.myCall = function(context,...args){
    context.fn = this;
    const result = context.fn(...args);
    delete context.fn;
    return result;
}

第三步:对传入null做处理

call函数中如果传入的是null则默认指向全局对象(MDN中call函数是这样描述的),所以给context赋值一个默认值

javascript 复制代码
Function.prototype.myCall = function(context,...args){
    context = context || window;
    context.fn = this;
    const result = context.fn(...args);
    delete context.fn;
    return result;
}

第四步:对fn属性名做唯一处理,怕属性名冲突

javascript 复制代码
Function.prototype.myCall = function(context,...args){
    context = context || window;
    const fn = Symbol('fn')
    context[fn] = this;
    const result = context[fn](...args);
    delete context[fn];
    return result;
}

使用示例

javascript 复制代码
const person = {
    name:"ling_wu"
}

function getName(age,sex){
    console.log(`name:${this.name},age:${age},sex:${sex}`)
}

//使用原生的call
getName.call(person,26,'男'); //输出:name:ling_wu,age:26,sex:男

//使用我们自己实现的myCall
getName.myCall(person,26,'男'); //输出:name:ling_wu,age:26,sex:男

📢 关于作者

嗨!我是头发秃头小宝贝,一名热爱技术分享的开发者,专注于Vue / 前端工程化 / 实战技巧 等领域。

如果这篇文章对你有所帮助,别忘了 点赞 👍收藏 ⭐关注 👏,你的支持是我持续创作的最大动力!

相关推荐
丷丩5 分钟前
MapLibre GL JS第31课:添加实时数据
javascript·gis·map·mapbox·maplibre gl js
小江的记录本24 分钟前
【JVM虚拟机】类加载机制:类加载全流程:加载→验证→准备→解析→初始化(附《思维导图》+《面试高频考点清单》)
java·jvm·spring boot·算法·安全·spring·面试
candyTong28 分钟前
Claude Code 每次调用 API 时,上下文是怎么"拼"出来的?
javascript·后端·架构
小林ixn33 分钟前
别再背“变量提升”了!深入编译执行,彻底搞懂 JavaScript 运行机制
javascript
用户8524950718435 分钟前
为什么变量能 未定义先使用?
javascript·程序员
Larcher1 小时前
JS 变量提升:代码没动,为什么执行顺序就变了?
前端·javascript·前端框架
yingyima1 小时前
MySQL 事件调度器速查:核心语法与实战代码
前端
GISer_Jing1 小时前
Claude Code多Agent架构深度剖析
前端·人工智能·架构·自动化
comphub1 小时前
comp-hub:让你的 Vue 业务组件真正"活"起来
前端
AI砖家1 小时前
Claude Code 跳过确认完全指南:让 AI 自己完成开发任务
前端·人工智能·python·ai编程·代码规范