JavaScript手写call,apply,bind方法

目录
  • 前言
  • 改写this实现思路
  • 前期准备
  • 手写call方法
  • 手写apply方法
  • 手写bind方法

前言

改变this指向在书写业务的时候经常遇到,我们经常采用以下方法进行改写

  • 使用作用声明变量存储this
  • 使用jJavaScript的原生方法call,apply,以及bind进行改写

第一种方法就不说了,就是一个变量存储的问题,主要说第二种如何实现的

call,bind,apply方法都是JavaScript原生的方法,挂载在Function原型上,使得所有函数都可以调用,今天我们来实现一下call,apply,bind吧

改写this实现思路

我们声明一个函数,函数接受俩个参数,第一个参数是接受改变的this指向,第二个是剩余参数,用于执行改变方法后传递的使用参数,在函数中里面声明一个变量用于存储传递过来的this,遵循this指向的准则,this是在执行的时候生成的,函数的准则是谁调用它就指向谁,传递过来的this是一个对象,然后在传递过来的this上面声明一个方法,声明的方法就是当前函数,该方法由传递过来的this进行调用执行,这样就改变了this指向问题,在把传递过来的参数除第一个外全部传递给声明的方法,我们来实现一下

前期准备

这些数据用于配合手写的方法,由于call,bind,apply他们都是需要把他们挂载到函数原型上,函数才能调用的到,我们这里也把他挂载到原型上

 var name = '前端';
        var age = "7";
        let obj = {
            name: '若水',
            age: '20'
        };

 function fnLog(arg) {
     let str = `我叫${this.name}今年${this.age}岁了`;
  console.log(str, '传递过来的', arg);
}

手写的call,bind,apply方法第一个参数都是改变的this指向,第二个参数是传递的参数

手写call方法

call方法是改变this后立即执行

  Function.prototype.myCall = function(content, ...arg) {
            // 创建一个变量接受改变的this,如果改变的this为空则采用window为默认this
            let newThis = content || window;
            /**newThis是一个对象,它是传递过来的this,在newThis上新增一个方法,这个方法就是
            当前函数 ,this就是指向的myCall,调用这个方法就可以改变this指向,因为函数是谁调用
            它this就指向谁
            **/
            newThis.fn = this;
           /**执行改变this方法,接受返回值,这里容易被误导,这里是把myCall传递过来的参数传递
            给了改变this的函数,由于不确定传递多少使用了剩余参数,这里也是 **/
            let result = newThis.fn(...arg);
            //删除传递过来this上声明的方法,防止出现问题
            delete newThis.fn;
            // 将执行结果返回
            return result;
        };
        //执行手写的call方法
        fnLog.myCall(obj, 'myCall');

手写apply方法

apply传递的参数只能传递数组

//如果arg没有传递参数,默认为空数组
      Function.prototype.myApply = function(content, arg = []) {
            // 判断传递参数是否为数组,如果不为数组则抛出错误,为数组则继续执行
            if (!Array.isArray(arg)) {
                throw '传递参数必须为数组';
            }
            // 创建一个变量接受改变的this,如果改变的this为空则采用window为默认this
            let newThis = content || window;
            // 给当前传递过来的this,添加一个方法,遵循this准则,谁调用就指向谁的原则
            newThis.fn = this;
            // 同mycall相同
            let result = newThis.fn(arg);
           //删除传递过来this上声明的方法,防止出现问题
            delete newThis.fn;
            // 将执行结果返回
            return result;
        };
        //执行
        fnLog.myApply(obj, [1, 212, 232]);

手写bind方法

bind不是立即执行而是返回一个新的函数

  Function.prototype.myBind = function(content, ...arg) {
            // 创建一个变量接受改变的this,如果改变的this为空则采用window为默认this
            let newThis = content || window;
        // 给当前传递过来的this,添加一个方法,遵循this准则,谁调用就指向谁的原则
            newThis.fn = this;
 // 由于bind不是立即执行,是返回一个函数作为bind方法执行的返回值,所以这里需要return出去一个函数
            return function() {
   // 然后在把执行当前的方法进行执行并把myBind接受剩余参数传递过去,接受返回值
                let result = newThis.fn(...arg);
                //删除当前的函数以及传递过来的this,防止出现问题
                delete newThis.fn;
                // 将执行结果返回
                return result;
            }
        };
       //调用bind函数,获取返回值,他的返回值是一个函数,所以我们需要在调用一下
        let f1 = fnLog.myBind(obj, 2);
        //调用bind的返回值
        f1()

到此这篇关于JavaScript手写call,apply,bind方法的文章就介绍到这了,更多相关JavaScript call内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Javascript中apply、call、bind的巧妙使用

    apply.call 在 javascript 中,call 和 apply 都是为了改变某个函数运行时的上下文(context)而存在的,换句话说,就是为了改变函数体内部 this 的指向. JavaScript 的一大特点是,函数存在「定义时上下文」和「运行时上下文」以及「上下文是可以改变的」这样的概念. 先来一个栗子: function fruits() {} fruits.prototype = { color: "red", say: function() { console

  • javascript中apply/call和bind的使用

    fun.apply(context,[argsArray]) 立即调用fun,同时将fun函数原来的this指向传入的新context对象,实现同一个方法在不同对象上重复使用. context:传入的对象,替代fun函数原来的this: argsArray:一个数组或者类数组对象,其中的数组参数会被展开作为单独的实参传给 fun 函数,需要注意参数的顺序. fun.call(context,[arg1],[arg2],[-]) 同apply,只是参数列表不同,call的参数需要分开一个一个传入.

  • 原生js如何实现call,apply以及bind

    1.实现call 步骤: 将函数设为对象的属性: 指定this到函数,并传入给定参数执行函数: 执行之后删除这个函数: 如果不传入参数,默认指向window: Function.prototype.mycall = function (context, ...args) { //判断是否为函数,如果不是函数,则报错 if (typeof this !== "function") { throw new Error("不是函数"); } context = conte

  • 一文搞懂JavaScript中bind,apply,call的实现

    目录 bind.call和apply的用法 bind call&apply 实现bind 实现call和apply 总结 bind.call和apply都是Function原型链上面的方法,因此不管是使用function声明的函数,还是箭头函数都可以直接调用.这三个函数在使用时都可以改变this指向,本文就带你看看如何实现bind.call和apply. bind.call和apply的用法 bind bind()方法可以被函数对象调用,并返回一个新创建的函数. 语法: function.bin

  • 使用JS简单实现apply、call和bind方法的实例代码

    目录 1.方法介绍 2.apply.call和bind方法的实现 2.1.apply的实现 2.2.call的实现 2.3.bind的实现 总结 1.方法介绍 apply.call和bind都是系统提供给我们的内置方法,每个函数都可以使用这三种方法,是因为apply.call和bind都实现在了Function的原型上(Function.prototype),而他们的作用都是给我们函数调用时显式绑定上this.下面先介绍一下它们的基本用法: apply方法:调用一个具有给定this值的函数,以及

  • JavaScript手写call,apply,bind方法

    目录 前言 改写this实现思路 前期准备 手写call方法 手写apply方法 手写bind方法 前言 改变this指向在书写业务的时候经常遇到,我们经常采用以下方法进行改写 使用作用声明变量存储this 使用jJavaScript的原生方法call,apply,以及bind进行改写 第一种方法就不说了,就是一个变量存储的问题,主要说第二种如何实现的 call,bind,apply方法都是JavaScript原生的方法,挂载在Function原型上,使得所有函数都可以调用,今天我们来实现一下c

  • JavaScript手写异步加法asyncAdd方法详解

    目录 前言 分析 asyncAdd 直观的基本要求 隐藏的考察点 — setTimeout & cb 隐藏的考察点 — async & await 实现 asyncAdd 具体实现 进行优化 抽离内层函数 缓存计算结果 前言 在掘金上发现一道既简单但个人觉得还挺有意思的一道题,题目如下: // 异步加法 function asyncAdd(a,b,cb){ setTimeout(() => { cb(null, a + b) }, Math.random() * 1000) } as

  • 「中高级前端面试」JavaScript手写代码无敌秘籍(推荐)

    1. 实现一个new操作符 new操作符做了这些事: 它创建了一个全新的对象. 它会被执行[[Prototype]](也就是__proto__)链接. 它使this指向新创建的对象.. 通过new创建的每个对象将最终被[[Prototype]]链接到这个函数的prototype对象上. 如果函数没有返回对象类型Object(包含Functoin, Array, Date, RegExg, Error),那么new表达式中的函数调用将返回该对象引用. function New(func) { va

  • JavaScript手写Promise核心原理

    目录 准备 完善 resolve/reject then 异步处理 链式调用 边界处理 catch 优化后完整代码 准备 首先,promise 有三种状态:pending fulfilled rejected; promise在实例化操作中, 有两个改变状态的方法,分别为resolve,reject; promise有很多方法,详情请见 mdn, 本篇文章先实现 promise的核心api: then和catch; 我们使用 es6 提供的 class 来实现 class MyPromise {

  • JavaScript手写数组的常用函数总结

    前言 在开发过程中,我们常常使用数组的一些 api 相关操作,其中包含 forEach . filter . find . findIndex . map . some . every . reduce . reduceRight 等函数方法. 今天,我们试试手写这些函数,实现数组这些函数方法.为了方便,直接在数组原型对象 prototype 上扩展. 本文 Githab 已上传,更多往期文章已分类整理. 正文 参数说明 callbackFn 回调函数 thisArg 执行 callbackFn

  • 万字详解JavaScript手写一个Promise

    目录 前言 Promise核心原理实现 Promise的使用分析 MyPromise的实现 在Promise中加入异步操作 实现then方法的多次调用 实现then的链式调用 then方法链式调用识别Promise对象自返回 捕获错误及 then 链式调用其他状态代码补充 捕获执行器错误 捕获then中的报错 错误与异步状态的链式调用 将then方法的参数变成可选参数 Promise.all方法的实现 Promise.resolve方法的实现 finally方法的实现 catch方法的实现 完整

  • 纯JavaScript手写图片轮播代码

    废话不多说了,直接给大家贴js代码实现手写图片轮播的代码了,代码非常简单,具体代码如下所示: <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>js图片轮播切换</title> <style type="text/css"> .imgCon{width: 400px;height: 400px;border: 2p

  • TensorFlow MNIST手写数据集的实现方法

    MNIST数据集介绍 MNIST数据集中包含了各种各样的手写数字图片,数据集的官网是:http://yann.lecun.com/exdb/mnist/index.html,我们可以从这里下载数据集.使用如下的代码对数据集进行加载: from tensorflow.examples.tutorials.mnist import input_data mnist = input_data.read_data_sets('MNIST_data', one_hot=True) 运行上述代码会自动下载数

  • JS面试中你不知道的call apply bind方法及使用场景详解

    目录 面试 三者区别 call apply. bind 总结 应用场景 简单实现 call apply bind 面试 面试官 :说一下 call apply bind 的三者区别吧? 我:啊.....这. 三者区别 call 我们先来看看 call call 函数接收两个参数 @params1 需要指向this @params2 需要携带的参数 就拿这段代码 来说 我调用 foo 函数 的时候 想去 执行obj 中的 eat函数 怎么办? 默认 我的 foo 里面什么页面有 怎么可能会打印 吃

  • 微信小程序实现简单手写签名组件的方法实例

    目录 背景: 需求: 效果 一.思路 二.实现 1. 页面与样式 2. 初始化 3. 点击时 4. 签名时 三.总结 背景: 在做项目过程中,需要在微信小程序中实现手写签名组件.在网上找了微信小程序手写签名实现,但是都是不太理想.在实际运用中,会因为实时计算较多的贝塞尔曲线而产生卡顿.效果不理想.所以退一步,不需要笔锋以及笔迹模拟效果.只需要简单的手写签名实现. 需求: 可以实现用户在微信小程序上手写签名. 需要组件化. 效果 一.思路 在微信小程序中,我们使用canvas组件实现.将用户的输入

随机推荐