实现 call、apply、bind

在之前一篇文章写了这三个参数的区别,但是其实面试更常考察如何实现。其实所有的原生函数的 polyfill 如何实现,只需要考虑 4 点即可:

  1. 基本功能
  2. 原型
  3. this
  4. 返回值

call

  1. call 的基本功能:

    call() 方法使用一个指定的 this 值和单独给出的一个或多个参数来调用一个函数。

  2. 原型

    不涉及原型链的转移,不用管
  3. this

    本质上,call 就是 this 的转移
  4. 返回值

简单实现:

Function.prototype.myCall = function(context = window, ...args) {
context.fn = this; // 先将fn挂在context上,
var res = context.fn(...args); // 然后通过context调用fn,使得fn中的this指向指到context上
delete context.fn; // 最后删除掉context上的fn
return res; // 返回原函数的返回值
};

上面为了简单,使用了 ES6 的剩余参数和展开语法,基本用这个回答面试官就好了。当然,如果不让使用剩余参数,那就只能使用eval或者new Function的字符串拼接大法了,可以参考这篇模板引擎

再就是 fn 可能会和 context 重名,整一个不会重名的 uniqueID 挂上去,执行完毕后删除。

apply

之前提过 apply 和 call 区别,只有一些入参和性能上的区别。直接上代码:

Function.prototype.myApply = function(context = window, args) {
context.fn = this; // 先将fn挂在context上,
var res = context.fn(...args); // 然后通过context调用fn,使得fn中的this指向指到context上
delete context.fn; // 最后删除掉context上的fn
return res; // 返回原函数的返回值
};

bind

bind 有点不一样,它会返回一个绑定了 this 的函数。

bind()方法创建一个新的函数,在 bind()被调用时,这个新函数的 this 被 bind 的第一个参数指定,其余的参数将作为新函数的参数供调用时使用。

Function.prototype.myBind = function(context, ...args) {
var fn = this; var newFn = function(...restArgs) {
// 使用call去调用fn,因为bind可能会bind一部分参数,所以把restArgs也传进去
return fn.call(context, ...args, ...restArgs);
}; return newFn;
};

上面的函数基本上覆盖了大部分场景,但是不能支持new调用——

绑定函数自动适应于使用 new 操作符去构造一个由目标函数创建的新实例。当一个绑定函数是用来构建一个值的,原来提供的 this 就会被忽略。不过提供的参数列表仍然会插入到构造函数调用时的参数列表之前。

如果直接使用我们上面所写的bind,就会返回

function Person(age, name) {
this.name = name;
this.age = age;
} var Age18Person = Person.myBind(null, 18); var a = {};
var Age20Person = Person.myBind(a, 20); var p18 = new Age18Person("test18"); // newFn {}
var p20 = new Age20Person("test20"); // newFn {}
// a {name: "test20", age: 20}
// window {name: "test18", age: 18}

显然,返回了以newFn生成的对象,并且,因为传入的是null,所以,对context的赋值转移到了window

这里需要判断是否被 new 调用,然后丢弃没用的 context。

Function.prototype.myBind = function(context, ...args) {
var fn = this; var newFn = function(...restArgs) {
// 如果是new构造,则使用new构造的实例
if (new.target) {
return fn.call(this, ...args, ...restArgs);
}
// 使用call去调用fn,因为bind可能会bind一部分参数,所以把restArgs也传进去
return fn.call(context, ...args, ...restArgs);
}; return newFn;
};

再次调用上面的new构造,发现实例的原型不是指向我们希望的 Person

var Age18Person = Person.myBind(null, 18);

var p18 = new Age18Person("test18"); // newFn {}

p instanceof Person; // false
p instanceof Age18Person; // false

记录一下原型链,再来一遍

Function.prototype.myBind = function(context, ...args) {
var fn = this; var newFn = function(...restArgs) {
// 如果是new构造,则使用new构造的实例
if (new.target) {
return fn.call(this, ...args, ...restArgs);
}
// 使用call去调用fn,因为bind可能会bind一部分参数,所以把restArgs也传进去
return fn.call(context, ...args, ...restArgs);
}; // 绑定原型链
newFn.prototype = this.prototype; return newFn;
};

但是这里还有个问题,如果改了Age18Personprototype,也会影响到Personprototype

所以,需要做一个中转——

Function.prototype.myBind = function(context, ...args) {
var fn = this; var newFn = function(...restArgs) {
// 如果是new构造,则使用new构造的实例
if (new.target) {
return fn.call(this, ...args, ...restArgs);
}
// 使用call去调用fn,因为bind可能会bind一部分参数,所以把restArgs也传进去
return fn.call(context, ...args, ...restArgs);
}; var NOOP = function() {}; // 绑定原型链
NOOP.prototype = this.prototype;
newFn.prototype = new NOOP(); return newFn;
};

这样基本上就算完成了,当然更推荐function-bind方案。

实现 call、apply、bind的更多相关文章

  1. call,apply,bind的用法

    关于call,apply,bind这三个函数的用法,是学习javascript这门语言无法越过的知识点.下边我就来好好总结一下它们三者各自的用法,及常见的应用场景. 首先看call这个函数,可以理解成 ...

  2. JavaScript中call,apply,bind方法的总结。

    why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之前,通常会有这些问题. var a = { user:"追梦子", fn:f ...

  3. call(),apply(),bind()与回调

    1.call(),apply(),bind()方法 JavaScript 中通过call或者apply用来代替另一个对象调用一个方法,将一个函数的对象上下文从初始的上下文改变为由 thisObj 指定 ...

  4. JS 的 call apply bind 方法

    js的call apply bind 方法都很常见,目的都是为了改变某个方法的执行环境(context) call call([thisObj[,arg1[, arg2[,   [,.argN]]]] ...

  5. javascript-this,call,apply,bind简述2

    上节我们一起研究了this这个小兄弟,得出一个结论,this指向调用this所在函数(或作用域)的那个对象或作用域.不太理解的朋友可以看看上节的内容,这次我们主要探讨一下call(),apply(), ...

  6. javascript-this,call,apply,bind简述1

    最近在系统的学习面向对象方面的知识,遇到的最大拦路虎就数this的指向,call,apply,bind函数的使用,单独抽出一天时间把这几个烦人的家伙搞定,去学习更深入的内容. 首先介绍一下this的一 ...

  7. call,apply,bind方法的总结

    why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之前,通常会有这些问题. var a = { user:"追梦子", fn:f ...

  8. JavaScript中call,apply,bind方法的总结

    原文链接:http://www.cnblogs.com/pssp/p/5215621.html why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之 ...

  9. call, apply,bind 方法解析

    call(), apply(),bind() 三者皆为Function的方法 call(),apply()的作用是调用方法,并改变函数运行时的context(作用上下文) bind() 的作用是引用方 ...

  10. JS中call,apply,bind方法的总结

    why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之前,通常会有这些问题. var a = { user: "小马扎", fn: ...

随机推荐

  1. 局部敏感哈希LSH(Locality-Sensitive Hashing)——海量数据相似性查找技术

    一. 前言     最近在工作中需要对海量数据进行相似性查找,即对微博全量用户进行关注相似度计算,计算得到每个用户关注相似度最高的TOP-N个用户,首先想到的是利用简单的协同过滤,先定义相似性度量(c ...

  2. WPF编程,C#中对话框自动关闭的一种方法(转载)

    本文原文链接:https://blog.csdn.net/qq_43307934/article/details/84933196———————————————— MessageBoxTimeout是 ...

  3. .NET Core 3.0 ,WTM 2.3.9发布

    .Net Core 3.0已经来了,WTM怎么可以落后呢.最新发布的WTM2.3.9版本已经支持.Net Core 3.0啦,现在在线生成项目的时候可以选择2.2和3.0两个版本.小伙伴们快来体验吧. ...

  4. 网络编程之winInet

    InternetGetConnectedState() 简介: 功能:检索本地系统的网络连接状态. 函数原型:BOOLAPI InternetGetConnectedState(            ...

  5. Powershell寻找域管在线服务器

    记录线下Powershell在域环境中对于服务器的信息收集 Powershell的脚本有很多,在内网渗透测试中不仅能扫,能爆,能转发,还能做更多的事情.我们常用的脚本有Powersploit,Empi ...

  6. 机器学习:eclipse中调用weka的Classifier分类器代码Demo

    weka中实现了很多机器学习算法,不管实验室研究或者公司研发,都会或多或少的要使用weka,我的理解是weka是在本地的SparkML,SparkML是分布式的大数据处理机器学习算法,数据量不是很大的 ...

  7. opencv::图像上采样和降采样

    图像金字塔概念 . 我们在图像处理中常常会调整图像大小,最常见的就是放大(zoom in)和缩小(zoom out),尽管几何变换也可以实现图像放大和缩小,但是这里我们介绍图像金字塔 . 一个图像金字 ...

  8. c++11::std::optional

    std::optional还有一个类似于智能指针的接口, 它可以显式转化为bool来表示std::optional是否有一个值. 指针的解引用操作符*和->都实现了, 但是没有std::bad_ ...

  9. IIS6.0使用冒号上传漏洞利用

    利用条件: 1.iis版本为6.0  2.上传文件名不会重命名 利用: 上传一个jpg木马图片 名字为:cs.asp:.jpg 注意是: 默认windows是不允许文件字含:(冒号)的 所以需要抓包后 ...

  10. git 拉取指定的远程分支(三种方式)

    直接拉取 git clone -b ants git@github.com:Ants-double/CareerJava.git git clone -b 远程分支名 仓库地址 本地已经有相关的仓库代 ...