当前位置: 首页 > news >正文

JS手写章节(1)—手写实现call、apply、bind

一、JS-call-apply-bind函数的基本使用

前言

在我们平常使用JS进行开发的过程中,有时会为某个函数的this的指向而苦恼,可能会因为拿不到想要的this而导致程序错误,为了解决这一问题,首先我们可以了解清楚this的四个绑定问题,即默认绑定,隐式绑定,显示绑定和new绑定,之前写过文章说过,懂了这个之后,我们可以判断清楚this的指向,而我们今天要说的即为显示绑定,直接使用call、apply、bind函数来粗暴的给想要的函数绑定我们给定的this。

1、call函数的基本使用

call函数是我们使用的比较多的函数,我们可以利用它来绑定this的值,call函数调用时候第一个参数要求我们传入要绑定的this,后面的参数要求我们传入函数的参数以逗号分隔,注意在调用call函数时候,原函数会直接执行一次。例如:
在这里插入图片描述

2、apply函数的基本使用

apply函数和call函数类似,唯一不同的是要求我们传递参数时候,以数组的形式传入,第一个参数依旧是我们需要绑定的this的值,注意调用apply函数时候,原函数也会立即执行
在这里插入图片描述
可以看出,与call调用得到的结果一样,其实我们可以利用apply函数参数是以数组形式传入的特点,配合扩展运算符,直接将参数以数组方式传递进去。
在这里插入图片描述

3、bind函数的基本使用

bind函数和call函数类似,传递的参数方式都是一样的,不一样的是bind函数调用不会立即执行,而会返回一个绑定了新的this的新函数。
在这里插入图片描述

我们可以看出,单纯调用bind函数没有任何反应,因为他不会立马执行,而是会返回一个绑定了新this的新函数。
在这里插入图片描述
图中可以看出,bind返回了一个新函数,参数即为在bind调用时传递的参数,如果bind调用时没有传递足够的参数,在返回的新函数中也可以传递参数,相当于放在原参数后面。

二、JS-函数的手写

前言

懂得了基本用法,我们可以自己使用一个call、apply、bind函数实行同样的效果。

1、call函数的手写

首先,我们知道call函数是所有function都可以调用的,因此我们可以在Function的原型对象上添加我们要实现的call函数,起名为cyj_call,即Function.prototype.cyj_call = function(thisArg,…argArray){},其次我们考虑如何将我们函数的this绑定到thisArg上?我们可以巧妙的使用使用this绑定规则中的隐式绑定,即thisArg.fn = fn;,这样我们的fn函数的this就指向thisArg了,那么还有一个问题,我们如何获取要原函数呢,还是用到了隐式绑定,我们要使用的时候肯定是类似这种使用方式的,fn.cyj_call(),那么这个时候由于cyj_call函数被fn调用,那么cyj_call函数里面的this便会隐式绑定到fn中,这时候我们在cyj_call函数里面的this就可以拿到原函数了,要注意的是我们可能绑定的this是一个字符串,那么这时候我们执行thisArg.fn = fn是报错的,因为字符串是不允许在上面添加fn属性的,我们可以执行Object(thisArg),这样便可以添加fn属性了,具体代码如下:

Function.prototype.cyj_call = function(thisArg, ...argArray) {
    thisArg = thisArg != undefined && thisArg != null ? Object(thisArg) : window;
    let fn = this;
    thisArg.fn = fn;
    let res = thisArg.fn(...argArray);
    delete thisArg.fn;
    return res;
};

在这里插入图片描述

2、apply函数的手写

apply函数的实现同理,只是在传递参数时候有些变化而已,通过call的手写,我们不难写出如下代码:

Function.prototype.cyj_bind = function(thisArg, argArray) {
    let fn = this;
    thisArg = thisArg != undefined && thisArg != null ? Object(thisArg) : window;
    argArray = argArray ? argArray : [];
    thisArg.fn = fn;
    let result = thisArg.fn(...argArray);
    delete thisArg.fn;
    return result;
};

在这里插入图片描述

3、bind函数的手写

bind函数的手写有些许不同,需要返回一个函数,并且返回的函数接受的参数要和bind调用时候的参数合在一起,传入原函数进行执行,然后返回结果。
不难写出以下代码:

Function.prototype.cyj_bind = function(thisArg, ...argArray) {
    thisArg = thisArg != undefined && thisArg != null ? Object(thisArg) : window;
    let fn = this;
    thisArg.fn = fn;
    return function(...args) {
        let res = thisArg.fn(...argArray, ...args);
        delete thisArg.fn;
        return res;
    };
};

在这里插入图片描述

三、结语

以上就是对JS中的call、apply、bind的手写实现,难度虽然不大,但是还是有些细节需要注意的,以上是我仓促写的代码,可能会有些没测试到的bug,但是总体来说应该没啥问题。以上就是JS-手写章节的第一篇,下一篇是手写实现Vue响应式原理,有问题欢迎私信!

相关文章:

  • JS - 获取剪切板内容 Clipboard API
  • 基于51单片机的心率体温监测报警系统(包括程序、仿真、原理图、流程图)
  • C++系列-谓词predicate
  • FreeRTOS-临界段保护及任务调度器的挂起和恢复
  • 《C++初始化列表陷阱:谨慎前行,避免潜在风险》
  • 基于SpringBoot+Vue的超市外卖管理系统
  • stable diffusion webUI之赛博菩萨【秋葉】——工具包新手安裝与使用教程
  • [FT]chatglm2微调
  • Linux按键输入实验-按键功能完善
  • 论文阅读:SOLOv2: Dynamic, Faster and Stronger
  • HashMap学习
  • 『python爬虫』xpath变化导致无法找到指定元素(持续更新中~)
  • Nginx配置ssl证书(https证书)
  • (附源码)计算机毕业设计SSM基于的扶贫产品展销平台
  • Google Earth Engine APP——影像条带色差、色调不均匀等现象解决方案Landsat5 NDWI Image Restoration APP
  • 【读书笔记】【Effective STL】函数子、函数子类、函数及其他
  • Linux磁盘LVM根目录扩容ubuntu--vg-ubuntu--lv
  • [MRCTF2020]Ezpop
  • 【day12】【洛谷算法题】-P5712Apples-刷题反思集[入门2分支结构]
  • PayPal/Stripe/Square 轮询收单系统
  • unity 位移贴图正弦波面
  • javaScript 防抖/节流,探索学习,对新手友好的内容
  • Servlet基础(1)
  • 【Svelte】-(4)If 条件判断语句 / Each 循环语句 / Await 异步处理块
  • 手把手实现一个进度条时钟,麻麻再也不用担心我把时间看茬了
  • java毕业设计潮购购物网站Mybatis+系统+数据库+调试部署
  • Verilog写状态机的三种描述方式之二段式
  • 【正点原子I.MX6U-MINI应用篇】1、编写第一个应用App程序helloworld
  • 【Vue】父子组件通信
  • AI 应用的全流程存储加速方案技术解析和实践分享
  • 云服务器的介绍
  • 《大数据分析技术》课程设计