前端JavaScript篇之实现call、apply 及 bind 函数

目录

  • 实现call、apply 及 bind 函数
    • 1. 实现call函数
    • 2. 实现apply函数
    • 3. 实现bind函数


实现call、apply 及 bind 函数

call、apply和bind函数都是用于改变函数中this指向的方法。它们的作用都是使函数能够在不同的对象上下文中运行。call方法和apply方法的作用类似,都是立即执行一个函数,并改变函数中this指向。bind方法和call、apply方法的作用也类似,都是改变函数中this指向。不同之处在于bind方法不会立即执行函数,而是返回一个新的函数,需要再次调用才能执行。

1. 实现call函数

call函数是JavaScript中的一个方法,它的作用是改变函数中this的指向,并立即执行该函数。call函数的作用是改变函数中this的指向,并且可以传递多个参数给函数。call函数的语法如下:

function.call(thisArg, arg1, arg2, ...)

其中,thisArg是指定的this值,arg1、arg2等是传递给函数的参数。如果需要传递多个参数,可以使用逗号分隔。

// 在全局作用域中定义自定义的myCall方法
Function.prototype.myCall = function (context) {// 判断调用对象if (typeof this !== 'function') {console.error('type error')}// 获取参数let args = [...arguments].slice(1),result = null// 判断 context 是否传入,如果未传入则设置为 windowcontext = context || window// 将调用函数设为对象的方法context.fn = this// 调用函数result = context.fn(...args)// 将属性删除delete context.fnreturn result
}

这段代码实现了一个自定义的call函数,它的作用和JavaScript内置的call函数类似,都是用来改变函数中this的指向。具体实现步骤如下:

  1. 判断调用对象是否为函数,如果不是则抛出错误。
  2. 使用展开运算符和slice函数获取传递给call函数的参数。
  3. 判断context是否传入,如果未传入则默认为window对象。
  4. 将当前函数设为context对象的一个属性。
  5. 调用context对象上的属性,并传入参数。
  6. 将context对象上的属性删除。
  7. 返回调用结果。

需要注意的是,在使用自定义的call函数时,需要确保传入的第一个参数是需要改变this指向的对象,后面的参数会作为函数调用时的参数传入。同时,为了避免修改context对象,我们需要在调用函数后将context对象上的属性删除。

下面是一个使用自定义的call函数的例子:

// 定义一个对象
let person = {name: '张三'
}// 定义一个函数
function sayHello(age) {console.log(`我叫${this.name},今年${age}岁。`)
}// 调用自定义的call函数
sayHello.myCall(person, 18) // 输出:我叫张三,今年18岁。

请添加图片描述

在这个例子中,我们定义了一个名为person的对象和一个名为sayHello的函数。使用自定义的call函数,将person对象作为第一个参数传入,改变sayHello函数中this的指向。同时,将18作为第二个参数传入,作为函数调用时的参数。最终输出了一句话,说明函数调用成功。

总结:通过实现自定义的call函数,我们可以更好地理解call函数的工作原理。call函数的作用是改变函数中this的指向,并立即执行该函数。我们可以将当前函数作为对象的一个属性来调用,从而改变函数中this的指向,使用apply函数获取传递给call函数的参数,并将它们传递给当前函数。最后删除对象上的属性。

2. 实现apply函数

apply是JavaScript中的一个函数方法,用于改变函数中this的指向,并立即执行该函数。apply函数和call函数的作用类似,不同之处在于apply函数的第二个参数是一个数组,这个数组中的元素会被作为参数传递给函数。apply函数和call函数的作用是一样的,只是传递参数的方式不同。apply函数的语法如下:

function.apply(thisArg, [argsArray])

其中,thisArg是需要改变this指向的对象,argsArray是一个数组,它的元素会作为函数调用时的参数传入。如果不需要传递参数,则可以传入一个空数组。

Function.prototype.myBind = function (context) {// 判断调用对象是否为函数if (typeof this !== 'function') {throw new TypeError('Error')}// 获取参数var args = [...arguments].slice(1),fn = thisreturn function Fn() {// 根据调用方式,传入不同绑定值return fn.apply(this instanceof Fn ? this : context, args.concat(...arguments))}
}

这段代码实现了一个自定义的myBind方法,用于改变函数中this的指向,并返回一个新的函数。myBind方法的作用和call、apply方法类似,不同之处在于它不会立即执行函数,而是返回一个新的函数,需要再次调用才能执行。

myBind方法的实现思路如下:

  1. 判断调用对象是否为函数,如果不是则抛出一个类型错误。
  2. 获取myBind方法传入的参数,其中第一个参数是需要改变this指向的对象,后面的参数是传递给函数的参数。
  3. 返回一个新的函数Fn,用于接收myBind方法调用时传入的参数。
  4. 在新的函数Fn中,根据调用方式,传入不同的绑定值,使用apply方法执行原函数fn,并将参数传递给原函数fn。
  5. 如果新函数Fn被当做构造函数使用,即通过new关键字调用,那么this指向新创建的对象,否则this指向myBind方法传入的第一个参数context。

下面是一个使用自定义的myBind方法的例子:

Function.prototype.myBind = function (context) {// 判断调用对象是否为函数if (typeof this !== 'function') {throw new TypeError('Error')}// 获取参数var args = [...arguments].slice(1),fn = thisreturn function Fn() {// 根据调用方式,传入不同绑定值return fn.apply(this instanceof Fn ? this : context, args.concat(...arguments))}
}// 定义一个对象
let person = {name: '张三'
}// 定义一个函数
function sayHello(age) {console.log(`我叫${this.name},今年${age}岁。`)
}// 使用自定义的myBind方法改变函数中this的指向,并返回一个新函数
let newFn = sayHello.myBind(person, 18)// 调用新函数
newFn() // 输出:我叫张三,今年18岁。

请添加图片描述

在这个例子中,我们定义了一个名为person的对象和一个名为sayHello的函数。使用自定义的myBind方法,将person对象作为第一个参数传入,改变sayHello函数中this的指向。同时,将18作为第二个参数传入,作为函数调用时的参数。调用myBind方法返回一个新的函数newFn,然后调用新函数newFn,输出了一句话,说明函数调用成功。

需要注意的是,使用自定义的myBind方法时,如果第一个参数不是一个对象,会导致函数调用失败。另外,如果新函数Fn被当做构造函数使用,即通过new关键字调用,那么this指向新创建的对象,否则this指向myBind方法传入的第一个参数context。

总结:myBind方法是用于改变函数中this指向的方法,并返回一个新的函数。myBind方法的实现思路是先判断调用对象是否为函数,然后获取myBind方法传入的参数,返回一个新的函数Fn,使用apply方法执行原函数fn,并将参数传递给原函数fn。如果新函数Fn被当做构造函数使用,那么this指向新创建的对象,否则this指向myBind方法传入的第一个参数context。在使用自定义的myBind方法时,需要注意第一个参数必须是一个对象,否则会导致函数调用失败。

3. 实现bind函数

bind是JavaScript中的一个函数方法,用于改变函数中this的指向,并返回一个新的函数,新函数中的this指向被绑定的对象,并且可以传递多个参数给函数。不会立即执行原函数。bind函数和call、apply函数的作用类似,不同之处在于bind函数返回一个新的函数,不会立即执行原函数。

bind函数的语法如下:

function.bind(thisArg[, arg1[, arg2[, ...]]])

其中,thisArg是需要改变this指向的对象,arg1、arg2等是传递给函数的参数。如果不需要传递参数,则可以省略后面的参数。

Function.prototype.myBind = function (context) {// 判断调用对象是否为函数if (typeof this !== 'function') {throw new TypeError('Error')}// 获取参数var args = [...arguments].slice(1),fn = thisreturn function Fn() {// 根据调用方式,传入不同绑定值return fn.apply(this instanceof Fn ? this : context, args.concat(...arguments))}
}

这段代码实现了一个自定义的myBind方法,用于改变函数中this的指向,并返回一个新的函数。myBind方法的作用和call、apply方法类似,不同之处在于它不会立即执行函数,而是返回一个新的函数,需要再次调用才能执行。

myBind方法的实现思路如下:

  1. 判断调用对象是否为函数,如果不是则抛出一个类型错误。
  2. 获取myBind方法传入的参数,其中第一个参数是需要改变this指向的对象,后面的参数是传递给函数的参数。
  3. 返回一个新的函数Fn,用于接收myBind方法调用时传入的参数。
  4. 在新的函数Fn中,根据调用方式,传入不同的绑定值,使用apply方法执行原函数fn,并将参数传递给原函数fn。
  5. 如果新函数Fn被当做构造函数使用,即通过new关键字调用,那么this指向新创建的对象,否则this指向myBind方法传入的第一个参数context。

下面是一个使用自定义的myBind方法的例子:

Function.prototype.myBind = function (context) {// 判断调用对象是否为函数if (typeof this !== 'function') {throw new TypeError('Error')}// 获取参数var args = [...arguments].slice(1),fn = thisreturn function Fn() {// 根据调用方式,传入不同绑定值return fn.apply(this instanceof Fn ? this : context, args.concat(...arguments))}
}// 定义一个对象
let person = {name: '张三'
}// 定义一个函数
function sayHello(age) {console.log(`我叫${this.name},今年${age}岁。`)
}// 使用自定义的myBind方法改变函数中this的指向,并返回一个新函数
let newFn = sayHello.myBind(person, 18)// 调用新函数
newFn() // 输出:我叫张三,今年18岁。

在这个例子中,我们定义了一个名为person的对象和一个名为sayHello的函数。使用自定义的myBind方法,将person对象作为第一个参数传入,改变sayHello函数中this的指向。同时,将18作为第二个参数传入,作为函数调用时的参数。调用myBind方法返回一个新的函数newFn,然后调用新函数newFn,输出了一句话,说明函数调用成功。

需要注意的是,使用自定义的myBind方法时,如果第一个参数不是一个对象,会导致函数调用失败。另外,如果新函数Fn被当做构造函数使用,即通过new关键字调用,那么this指向新创建的对象,否则this指向myBind方法传入的第一个参数context。

总结:myBind方法是用于改变函数中this指向的方法,并返回一个新的函数。myBind方法的实现思路是先判断调用对象是否为函数,然后获取myBind方法传入的参数,返回一个新函数Fn,使用apply方法执行原函数fn,并将参数传递给原函数fn。如果新函数Fn被当做构造函数使用,那么this指向新创建的对象,否则this指向myBind方法传入的第一个参数context。在使用自定义的myBind方法时,需要注意第一个参数必须是一个对象,否则会导致函数调用失败。

总结:
call、apply和bind函数是JavaScript中用于改变函数中this指向的方法。它们的作用都是改变函数中的this指向,使函数能够在不同的对象上下文中运行。

call和apply方法的作用类似,都是立即执行一个函数,并改变函数中this指向。不同之处在于传递参数的方式不同,call方法是将参数一个一个地传递给函数,而apply方法是将参数作为一个数组传递给函数。

bind方法和call、apply方法的作用也类似,都是改变函数中this指向。不同之处在于bind方法不会立即执行函数,而是返回一个新的函数,需要再次调用才能执行。

持续学习总结记录中,回顾一下上面的内容:
call、apply和bind函数都是用于改变函数中this指向的方法。它们的作用都是使函数能够在不同的对象上下文中运行。call方法和apply方法的作用类似,都是立即执行一个函数,并改变函数中this指向。bind方法和call、apply方法的作用也类似,都是改变函数中this指向。不同之处在于bind方法不会立即执行函数,而是返回一个新的函数,需要再次调用才能执行。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/676957.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

多元回归分析:理论与应用

多元回归分析是一种统计方法,用于研究两个或多个自变量(解释变量)与一个因变量(响应变量)之间的关系。这种分析允许研究者评估多个因素对结果变量的影响,是社会科学、经济学、生物医学和工程等多个领域中常…

【doghead】uv_loop_t的创建及线程执行

worker测试程序,类似mediasoup对uv的使用,是one loop per thread 。创建一个UVLoop 就可以创建一个uv_loop_t Transport 创建一个: 试验配置创建一个: UvLoop 封装了libuv的uv_loop_t ,作为共享指针提供 对uv_loop_t 创建并初始化

鸿蒙(HarmonyOS)项目方舟框架(ArkUI)之Toggle组件

鸿蒙(HarmonyOS)项目方舟框架(ArkUI)之Toggle组件 一、操作环境 操作系统: Windows 10 专业版、IDE:DevEco Studio 3.1、SDK:HarmonyOS 3.1 二、Toggle组件 组件提供勾选框样式、状态按钮样式及开关样式。 子组件 仅当Toggl…

【MySQL】数据库基础 -- 详解

一、什么是数据库 存储数据用文件就可以了,为什么还要弄个数据库? 一般的文件确实提供了数据的存储功能,但是文件并没有提供非常好的数据(内容)的管理能力(用户角度)。 文件保存数据有以下几个缺点&…

无心剑中译佚名《春回大地》

The Coming of Spring 春回大地 I am coming, little maiden, With the pleasant sunshine laden, With the honey for the bee, With the blossom for the tree. 我来啦,小姑娘 满载着欣悦的阳光 蜂儿有蜜酿 树儿有花绽放 Every little stream is bright, All …

Windows 安装 Linux子系统,并为子系统设置图形化界面

安装WSL 在控制面板中打开下面的选项: 执行下面的命令,更新到WSL2版本,并在以后创建子系统的时候默认采用WSL2的版本: wsl --update wsl --set-default-version 2在Window上安装连接工具: 在Window上下载VcXsrv&…

备战蓝桥杯---动态规划之经典背包问题

看题: 我们令f[i][j]为前i个物品放满容量为j的背包的最大价值。 f[i][j]max(f[i-1][j],f[i-1][j-c[i]]w[i]); 我们开始全副成负无穷。f[0][0]0;最后循环最后一行求max; 负无穷:0xc0c0c0c0;正无穷:0x3f3f3f3f 下面是v12,n6的图示&#xff…

深搜问题:素数圆环

祝大家新年快乐,今天给大家带来龙年第一道题 时间限制:1秒 内存限制:128M 题目描述 如图所示为一个由n个圆圈构成的圆环。将自然数1,2,...,n放入圆圈内,并且要求任意两个相邻的圆圈内…

自动化UI,API和DevOps测试架构设计与实现

自动化测试是软件开发过程中的重要环节,它可以提高测试效率、减少人工测试的工作量。本文将介绍自动化测试架构的设计原则和实现方法,以帮助读者理解如何构建一个可靠、可扩展和易于维护的自动化测试框架。 1. 什么是自动化测试? - 解释了…

二阶系统的迹-行列式平面方法(trace-determinant methods for 2nd order system)

让我们再次考虑二阶线性系统 d Y d t A Y \frac{d\mathbf{Y}}{dt}A\mathbf{Y} dtdY​AY 我们已经知道,分析这种二阶系统。最主要的是注意它的特征值情形。 (此处没有重根的情形,所有是partial) 而特征值,也就是系…

Electron+Vue实现仿网易云音乐实战

前言 这个项目是我跟着官方文档的那个Electron入门教程大致跑了一遍,了解了下Electron开发流程之后的实战项目,所以中间应该是会有很多写法不是很规范,安全性有可能也没考虑到,可实现的各种api也不是很了解,适合初学者。 必须感谢 https://github.com/Binaryify/NeteaseC…

Python 数据可视化之山脊线图 Ridgeline Plots

文章目录 一、前言二、主要内容三、总结 🍉 CSDN 叶庭云:https://yetingyun.blog.csdn.net/ 一、前言 JoyPy 是一个基于 matplotlib pandas 的单功能 Python 包,它的唯一目的是绘制山脊线图 Joyplots(也称为 Ridgeline Plots&…

【C语言】一道相当有难度的指针某大厂笔试真题(超详解)

这是比较复杂的题目,但是如果我们能够理解清楚各个指针代表的含义,画出各级指针的关系图,这道题就迎刃而解了。 学会这道笔试题,相信你对指针的理解,对数组,字符串的理解都会上一个档次。 字符串存储使用的…

MFC实现遍历系统进程

今天我们来枚举系统中的进程和结束系统中进程。 认识几个API 1)CreateToolhelp32Snapshot 用于创建系统快照 HANDLE WINAPI CreateToolhelp32Snapshot( __in DWORD dwFlags, //指定快照中包含的系统内容__in DWORD th32P…

【华为 ICT HCIA eNSP 习题汇总】——题目集15

1、(多选)以下 eSight 网管支持的远程告警通知方式包括()。 A、邮件 B、语音 C、视频 D、短信 考点:网络运维 解析:(AD) eSight 网管支持的远程告警通知方式主要包括邮件和短信通知&…

【网站项目】031网络游戏公司官方平台

🙊作者简介:拥有多年开发工作经验,分享技术代码帮助学生学习,独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。🌹赠送计算机毕业设计600个选题excel文件,帮助大学选题。赠送开题报告模板&#xff…

网络编程-Socket套接字

目录 1.网络编程 1.1定义与图解 1.2基本概念 (1)发送端和接收端 (2)请求和响应 (3)客户端和服务端 2.Socket套接字 2.1定义 2.2分类 (1)流套接字 (2&#xff…

攻防世界——re2-cpp-is-awesome

64位 我先用虚拟机跑了一下这个程序,结果输出一串字符串flag ——没用 IDA打开后 F5也没有什么可看的 那我们就F12查看字符串找可疑信息 这里一下就看见了 __int64 __fastcall main(int a1, char **a2, char **a3) {char *v3; // rbx__int64 v4; // rax__int64 v…

知识图谱与图神经网络融合:构建智能应用的新前沿

目录 前言1 知识图谱表示学习1.1 典型模型1.2 下游任务 2 图神经网络与知识图谱表示学习2.1 Compgcn:合成图卷积模型2.2 知识图谱嵌入在归纳设置下的推进 3 图神经网络与知识图谱构建3.1 关系抽取的进阶应用3.2 结构信息补全与知识图谱的完整性 4 图神经网络与知识图…

JavaScript综合练习4

JavaScript 综合练习 4 1. 案例演示 2. 代码实现 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title&…