对比学习 apply, call, 和 bind 👀👀👀

前言

大家好 , 我是一名在校大二学生 , 今天学习apply, call, 和 bind , 我决定将三者对比学习 , 分别从 :

  • 相同点
  • 差异点
  • 易错点

横向对比

在 JavaScript 开发中,apply, call, 和 bind 是非常常用的函数方法,它们都用于改变函数的 this 上下文。本文将详细介绍这三者的相同点、区别以及易错点,并通过代码示例来加深理解。

相同点

  1. 改变 this 上下文 :这三个方法都可以改变函数调用时的 this 值。
  2. 立即执行 vs 延迟执行applycall 会立即执行函数,而 bind 会返回一个新函数,可以在之后调用。
代码示例
js 复制代码
function greet(name) {
    console.log(`Hello, ${name}! I am ${this.name}`);
}

const person = { name: 'Alice' };

// 使用 call
greet.call(person, 'Bob'); // Hello, Bob! I am Alice

// 使用 apply
greet.apply(person, ['Charlie']); // Hello, Charlie! I am Alice

// 使用 bind
const greetAlice = greet.bind(person);
greetAlice('Dave'); // Hello, Dave! I am Alice

差异点

  1. 参数传递方式
    • call:参数以逗号分隔的形式传递。
    • apply:参数以数组形式传递。
    • bind:参数以逗号分隔的形式传递,但返回的是一个新函数,可以在之后调用。
代码示例
javascript 复制代码
function sum(a, b, c) {
    return a + b + c;
}

const numbers = [1, 2, 3];

// 使用 call
console.log(sum.call(null, 1, 2, 3)); // 6

// 使用 apply
console.log(sum.apply(null, numbers)); // 6

// 使用 bind
const sum12 = sum.bind(null, 1, 2);
console.log(sum12(3)); // 6
  1. 返回值
    • callapply:返回函数的执行结果。
    • bind:返回一个新函数,可以在之后调用。
代码示例
js 复制代码
function getGreeting(name) {
    return `Hello, ${name}! I am ${this.name}`;
}

const person = { name: 'Alice' };

// 使用 call
const greetingCall = getGreeting.call(person, 'Bob');
console.log(greetingCall); // Hello, Bob! I am Alice

// 使用 apply
const greetingApply = getGreeting.apply(person, ['Charlie']);
console.log(greetingApply); // Hello, Charlie! I am Alice

// 使用 bind
const getGreetingAlice = getGreeting.bind(person);
const greetingBind = getGreetingAlice('Dave');
console.log(greetingBind); // Hello, Dave! I am Alice

易错点

  1. this 上下文的默认值

    • 在严格模式下,this 默认为 undefined
    • 在非严格模式下,this 默认为全局对象(浏览器中是 window,Node.js 中是 global)。
代码示例
scss 复制代码
function logThis() {
    console.log(this);
}

// 严格模式
(function() {
    'use strict';
    logThis.call(); // undefined
})();

// 非严格模式
logThis.call(); // window 或 global
  1. bind 返回的新函数的 this 上下文
  • bind 返回的新函数的 this 上下文是固定的,即使使用 callapply 也无法改变。
代码示例
javascript 复制代码
function logThis() {
    console.log(this);
}

const boundLogThis = logThis.bind({ name: 'Alice' });

boundLogThis.call({ name: 'Bob' }); // { name: 'Alice' }
  1. apply call 的性能
  • apply 需要将参数数组展开,性能略低于 call
代码示例
js 复制代码
function sum(a, b, c) {
    return a + b + c;
}

const numbers = [1, 2, 3];

console.time('call');
for (let i = 0; i < 1000000; i++) {
    sum.call(null, 1, 2, 3);
}
console.timeEnd('call'); // call: 10ms

console.time('apply');
for (let i = 0; i < 1000000; i++) {
    sum.apply(null, numbers);
}
console.timeEnd('apply'); // apply: 15ms

总结

apply, call, 和 bind 都是改变函数 this 上下文的强大工具,但它们在参数传递方式、立即执行 vs 延迟执行、返回值等方面有所不同。理解这些差异和易错点,可以帮助我们在实际开发中更灵活地使用这些方法,提高代码的可读性和可维护性。

希望本文对你有所帮助!如果有任何疑问或建议,欢迎在评论区留言交流。

相关推荐
华玥作者14 小时前
[特殊字符] VitePress 对接 Algolia AI 问答(DocSearch + AI Search)完整实战(下)
前端·人工智能·ai
Mr Xu_15 小时前
告别冗长 switch-case:Vue 项目中基于映射表的优雅路由数据匹配方案
前端·javascript·vue.js
前端摸鱼匠15 小时前
Vue 3 的toRefs保持响应性:讲解toRefs在解构响应式对象时的作用
前端·javascript·vue.js·前端框架·ecmascript
sleeppingfrog15 小时前
zebra通过zpl语言实现中文打印(二)
javascript
lang2015092815 小时前
JSR-340 :高性能Web开发新标准
java·前端·servlet
好家伙VCC16 小时前
### WebRTC技术:实时通信的革新与实现####webRTC(Web Real-TimeComm
java·前端·python·webrtc
未来之窗软件服务16 小时前
未来之窗昭和仙君(六十五)Vue与跨地区多部门开发—东方仙盟练气
前端·javascript·vue.js·仙盟创梦ide·东方仙盟·昭和仙君
baidu_2474386116 小时前
Android ViewModel定时任务
android·开发语言·javascript
嘿起屁儿整16 小时前
面试点(网络层面)
前端·网络
VT.馒头17 小时前
【力扣】2721. 并行执行异步函数
前端·javascript·算法·leetcode·typescript