手写实现call() apply() bind()函数,附有详细注释,包含this指向、arguments讲解

news2024/12/23 17:47:04

手写实现call() apply() bind()函数是很经典的问题,但是能掰扯清楚的文章确实不算多,于是笔者才决定写下本文,希望能给读者带来一些启发,如有错误欢迎指正。

目录

补充知识

函数中的this指向

 类数组对象arguments

call()

原理:

详细实现:

验证:

apply()

原理:

详细实现:

 验证:

bind()

原理:

详细实现:

验证:


补充知识

在往下看具体的实现之前,我们先要了解一些前置补充知识

函数中的this指向

函数中的this指向是在函数被调用的时候确定的,也就是执行上下文被创建时确定的。

在一个执行上下文中,this由调用者提供,调用函数的方式来决定。

1.方法调用模式

哪个对象调用函数(object.method()),this就指向哪个对象

let obj={
    a:1,
    b:2,
}
obj.test=function(){
    let a=3
    console.log(this,this.a)
}
obj.test()
//结果为 { a: 1, b: 2, test: [Function: test] } 1
//test的this指向obj

2.独立调用模式

独立调用时,指向window(严格模式下指向undefined)

// 例子1
window.a=111
var test1 =function(){
    let a=123
    console.log(this.a)
}
test1() // 输出结果为111,说明此时test1的this指向是全局的window

// 例子2
window.a=111
var obj = {
        a:222
}

obj.test2=function(){
    let a=333;
    console.log(this.a,'第一个')
    let func=function(){
        console.log(this.a,'第二个')
    }
    func()
}

obj.test2()
// 输出结果为:
// ’222 第一个‘ 说明this指向是obj,因为test2是方法调用
// ‘111 第二个’ 说明this指向是window,因为func是独立调用

3.构造函数模式

js中,我们通过new关键词来调用构造函数,此时this会绑定在该实例对象

window.a=111
test3=function(){
    this.a=444
}
test3()
console.log(window.a) 
//结果为‘444’,说明this指向的是windows,因为上面test3的调用模式是独立调用

let newObj= new test3()
console.log(newObj.a) 
// 结果为‘444’,说明this指向的是newObj,因为上面的test3的调用是作为构造函数的形式调用

 类数组对象arguments

arguments只在函数中存在(箭头函数除外)的伪数组(具有length,可以通过下标访问,但是不具有数组的方法, 比如push(),pop()等数组常用的方法),存储了我们传入的所有形参

例子

function testArgu(){
    console.log(arguments);
}
testArgu('a','b',123)

运行结果

我们可以看到,我们传的参数都在arguments中了 

call()

原理

当函数执行call方法的时候,实际上是把函数放到call()的第一个参数的某个属性上,然后再通过合格属性来执行函数

func.call(ctx,arg1,...)

//等价于以下代码
ctx.fn=func;
ctx.fn(arg1,....)

详细实现

Function.prototype.myCall=function(context,...args){
    //对this进行类型判断,如果不是function类型,就报错
    //this应该指向的是调用myCall函数的对象(function也属于对象object类型)
    //因为myCall的调用模式是上文提到的‘方法调用模式’
    if(typeof this != 'function'){
        throw new TypeError('type error')
    }

    // 不传的话,默认上下文是window
    var context = context || window
    
   // 假如context上面有fn属性的时候,会出现冲突
   // 所以当context上面有fn属性的时候,要先保存一下
    var temp = null
    if (context.fn) {
        temp = context.fn
    }

    // 给context创建一个fn属性,并将值设置为需要调用的函数(即this)
    context.fn = this

    //调用函数
    const res = context.fn(...args)

    // 删除context对象上的fn属性
    if (temp) {
        context.fn = temp
    } else {
        delete context.fn
    }

    // 返回运行结果
    return res
}

验证

let num=1;
let obj={
    num:2,
    fn:'this is obj.fn'
}


function test(a,b,c,d){
    console.log(this.num,'test参数:',a,b,c,d)
}

// 调用myCall函数
test.myCall(obj,4,3,2,1)

// 检查obj本身的fn是否被修改
console.log(obj.fn)

以上验证代码运行结果为:

说明手写的myCall方法可以修改this指向,并且obj本身的fn未被修改

apply()

原理:

基本原理和call类似,区别就是对参数对处理不同:

call方法接受的参数是一个参数列表,而apply方法接受的是一个包含多个参数的数组。

这里我们就可以用到上文说的类数组对象arguments来处理参数了

详细实现:

// 和myCall的不同之处1:参数
Function.prototype.myApply=function(context){
    if(typeof this!== 'function'){
        throw new TypeError('type error')
    }
    var context = context || window
    var temp = null
    if (context.fn) {
        temp = context.fn
    }
    context.fn = this

    let res;
    // 和myCall的不同之处2:参数的处理
    // 判断第二个参数是否存在
    if (arguments[1]) {
        res = context.fn(...arguments[1])
    }else{
        res = context.fn()
    }

    // 删除context对象上的fn属性
    if (temp) {
        context.fn = temp
    } else {
        delete context.fn
    }

    // 返回运行结果
    return res
}

 验证:

let num=1;
let obj={
    num:2,
    fn:'this is obj.fn'
}

function test(a,b,c,d){
    console.log(this.num,'test参数:',a,b,c,d)
}

// 调用myCall函数
test.myApply(obj,[1,2,3,4])

// 检查obj本身的fn是否被修改
console.log(obj.fn)

 运行结果为:

 

bind()

原理:

bind需要考虑一种情况,就是bind返回的函数作为构造函数使用的时候,bind绑定的this失效,但是参数依旧有效。

如何区分bind是正常使用还是当构造函数使用呢?根据this判断就行了。因为函数的this指向取决于如何调用(上文讲到过)。

1.当构造函数的时候,this指向新建的实例对象(此时this的prototype在该构造函数上)。

2.正常使用,this指向window

详细实现:

Function.prototype.myBind=function(context){
    if(typeof this!== 'function'){
        throw new TypeError('type error')
    }
    // 获取参数
    var args0 =[...arguments].slice(1);
    // 保存this,如果作为构造函数使用,此时this会指向实例
    fn=this;

    //返回更改this指向的函数
    return function Fn(...args){
        //如果是new的形式来使用绑定函数的
        if(this instanceof Fn) 
            return new fn( ...args0,...args)
        //如果是普通函数的形式来使用绑定函数的
        else 
            return fn.call(context, ...args0,...args);
    }
}

验证:

function Point(x, y) {
  this.x = x;
  this.y = y;
}

// 情况1:正常调用bind函数
var testObj = {};
var YAxisPoint = Point.myBind(testObj, 0 );
YAxisPoint(1)
console.log(testObj)

// 情况2:bind返回函数作为构造函数
// 此时之前绑定的指向testObj的this会失效,会重新指向新的对象实例,但是参数会继续有效
let newObj=new YAxisPoint(2);
console.log(newObj)

运行结果为

符合预期

 

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

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

相关文章

Leedcode19. 删除链表的倒数第 N 个结点

给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。 输入:head [1,2,3,4,5], n 2 输出:[1,2,3,5] 示例 2: 输入:head [1], n 1 输出:[] 示例 3: 输入&#xff1…

MPI之持久化通信句柄与非持久化通信句柄

MPI_Isend & MPI_Send 创建临时通信句柄 在前面的文章中举了例子,我们使用MPI_Isend接口发送数据时,有个传出参数request,该参数是创建的通信句柄, 实际上该句柄是一个临时句柄,即只用于一次性发送数据的场景&…

uniapp iOS打包证书申请流程——mac

如何在 Mac 创建 iOS 打包证书? 文章目录 如何在 Mac 创建 iOS 打包证书?会员 VS 非会员权限步骤添加设备创建标识符生成证书生成描述文件 前提: Mac 电脑Apple ID 申请Apple ID成为开发者 developer 注意: 登录 Apple ID 成为开…

2023年数字孪生行业研究报告

第一章 行业概况 1.1 定义 数字孪生(Digital Twin)是一种先进的建模技术,它通过创建一个物理实体的虚拟复制品,以实时模拟、预测和优化实体的行为和性能。这个虚拟模型会同步收集和分析来自其物理对应物的数据,从而提…

【DRONECAN】(三)WSL2 及 ubuntu20.04 CAN 驱动安装

【DRONECAN】(三)WSL2 及 ubuntu20.04 CAN 驱动安装 前言 这一篇文章主要介绍一下 WSL2 及 ubuntu20.04 CAN 驱动的安装,首先说一下介绍本文的目的。 大家肯定都接触过 ubuntu 系统,但是我们常用的操作系统都是 Windows&#x…

JavaSE基础(2)

1 方法的使用 思维导图 1.1 什么是方法 方法就是一个代码片段. 类似于 C 语言中的 “函数”。方法存在的意义(不要背, 重在体会): 是能够模块化的组织代码(当代码规模比较复杂的时候).做到代码被重复使用, 一份代码可以在多个位置使用.让代码更好理解更简单.直接调用现有方法…

基于金枪鱼群算法优化的BP神经网络(预测应用) - 附代码

基于金枪鱼群算法优化的BP神经网络(预测应用) - 附代码 文章目录 基于金枪鱼群算法优化的BP神经网络(预测应用) - 附代码1.数据介绍2.金枪鱼群优化BP神经网络2.1 BP神经网络参数设置2.2 金枪鱼群算法应用 4.测试结果:5…

肠道微生物群肾衰竭

编者推荐 该研究应用多组学分析(代谢组学分析微生物组学分析)分析了人类ESRD肠道微生物组组成、尿毒症毒素和肾衰竭之间的关系,使用独立队列和无菌动物模型对多组学结果及研究提出的ESRD机制假设进行验证,首次从肠道微生物的角度…

qt第一天

#include "widget.h" #include "ui_widget.h" #include "QDebug" Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);this->resize(QSize(800,600)); //使用匿名对象,调用重…

嵌入式Linux开发实操(十五):nand flash接口开发

# 前言 flash memory,分NAND和NOR: 如果说nor flash有个特点就是能执行代码,NOR并行接口具有地址和数据总线,spi flash更是主要用于存储代码,SPI(或QSPI)NOR代码可就地执行(XiP),一般系统要求flash闪存提供相对较高的频率和数据缓存的clocking。而nand flash主要用于…

QT Creator工具介绍及使用

一、QT的基本概念 QT主要用于图形化界面的开发, QT是基于C编写的一套界面相关的类库,如进程线程库,网络编程的库,数据库操作的库,文件操作的库等。 如何使用这个类库:类库实例化对象(构造函数) --> 学习…

Django静态文件媒体文件文件上传

文章目录 一、静态文件和媒体文件1.在django中使用静态文件实践2.在django中使用媒体文件 二、文件上传单文件上传实践多文件上传 一、静态文件和媒体文件 媒体文件: 用户上传的文件,叫做media 静态文件:存放在服务器的css,js,image,font等 叫做static1.在django中…

【Flutter】使用Android Studio 创建第一个flutter应用。

前言 首先下载好 flutter sdk和 Android Studio。 FlutterSDK下载 Android Studio官网 配置 我的是 windows。 where.exe flutter dart查看flutter安装环境。 如果没有,自己在环境变量的path添加下flutter安装路径。 在将 Path 变量更新后,打开一个…

QTday1(第一个QT界面、常用类与组件)

一、Xmind整理: Assistant帮助文档的使用: 设计师界面的介绍: 各文件之间调用方式: 二、上课笔记整理: 1.第一个QT界面 ①创建自定义类时需要指定父类 ②第一个界面的相关操作 #include "mainwindow.h"…

比Python快3.5万倍的Mojo融资7亿,LLVM之父:不会威胁到Python,该恐惧的应该是C++

近日,Modular AI 公司宣布成功融资 1 亿美元(约 7.29 亿人民币),据称这是继去年 3000 万美元融资之后的第二轮融资。 Modular AI 称他们未来的愿景是通过 AI 引擎和 Mojo 为全球开发者提供 AI 基础设施。 Modular AI 是 Chris La…

maven本地安装jar包install-file,解决没有pom的问题

背景: 公司因为权限问题,没有所有的代码,内部maven还在搭建,所以需要拿到同事的jar包,本地install: mvn install:install-file -DgroupIdcom..framework -DartifactIdcloud-api -Dversion1.0.0-SNAPSHOT …

Blender 3D建模要点

3d模型可以为场景的仿真模拟带来真实感,它还有助于更轻松地识别场景中的所有内容。 例如,如果场景中的所有对象都是简单的形状,如立方体和圆形,则很难在仿真中区分对象。 1,碰撞形状与视觉形状 像立方体和球体这样的简单形状,通常被称为“基本体”,通常用作碰撞块。 与…

js优雅的统计字符串字符出现次数

题目如下 统计一串字符串中每个字符出现的频率 示例字符串 let str asdfasqwerqwrdfafafasdfopasdfopckpasdfassfd小白写法 let str asdfasqwerqwrdfafafasdfopasdfopckpasdfassfdlet result {}; for (let i 0; i < str.length; i) {if (result[str[i]]) {result[str[…