call方法

基础版, 只能修改指向,不能传参

Function.prototype.myCall = function(context) {
// 获取调用者,这里为bar
context.fn = this;
// 运行函数
context.fn();
// 删除缓存
delete context.fn;
} let foo = {
value: 1
} function bar() {
console.log(this.value);
} bar.myCall(foo);
// 1

eval版本

Function.prototype.myCall = function(context) {
// 获取调用者,这里为sayHi
// 将无调用者的情况转换为有调用者的情况
// 有调用者那么函数内部的this就指向调用者
context.fn = this;
var len = arguments.length;
// 保存传递的参数
var args = Array(len - 1);
for (var i = 1; i < len; i++) {
// 这里只是把获取参数的字符串拼接了,供eval调用
args[i - 1] = 'arguments[' + i + ']';
}
// 不直接调用而用eval是因为参数个数不一定
// 在这里相当于是在执行context.fn(arguments[1],arguments[2]);
eval('context.fn('+ args +')');
delete context.fn;
} let person = {
name: 'Wango'
} function sayHi(age, sex) {
console.log(this.name, age, sex)
} sayHi.myCall(person, 24, 'male');
// Wango 24 male

ES6 版本

Function.prototype.myCall = function(context) {
context.fn = this;
// 将arguments转换为数组并调用slice方法去除第一个参数
// 再使用...将数组打散
context.fn(...Array.from(arguments).slice(1));
delete context.fn;
} let person = {
name: 'Wango'
} function sayHi(age, sex) {
console.log(this.name, age, sex);
} sayHi.myCall(person, 24, 'male');
// Wango 24 male

ES6 升级版

Function.prototype.myCall = function(context, ...args) {
context.fn = this;
// 相比使用arguments,这里只是用了ES6的剩余参数
context.fn(...args);
delete context.fn;
} let person = {
name: 'Wango'
} function sayHi(age, sex) {
console.log(this.name, age, sex);
} sayHi.myCall(person, 24, 'male');
// Wango 24 male

最终版

之前的版本如果传入的对象原本的fn属性或方法会被覆盖,然后被删除;而且传入第一个参数如果不是对象,会报错,所以还需要一些容错处理

// 保存一个全局变量作为默认值
const root = this; Function.prototype.myCall = function(context, ...args) {
if (typeof context === 'object') {
// 如果参数是null,使用全局变量
context = context || root;
} else {
// 参数不是对象的创建一个空对象
context = Object.create(null);
}
// 使用Symbol创建唯一值作为函数名
let fn = Symbol();
context[fn] = this;
context[fn](...args);
delete context[fn];
} let person = {
name: 'Wango',
fn: function() {
console.log(this.name);
}
} function sayHi(age, sex) {
console.log(this.name, age, sex);
} sayHi.myCall(person, 24, 'male');
// Wango 24 male
sayHi.myCall(null, 24, 'male');
// undefined 24 male
sayHi.myCall(123, 24, 'male');
// undefined 24 male
// 原函数不受影响
person.fn();
// Wango

call的实现最核心的部分就是将没有调用者的情况转换为有调用者的情况,函数内部的this自然就指向调用者

apply方法

apply的实现思路和call方法是一样的,只是只接收两个参数,第二个参数为类数组

const root = this;

Function.prototype.myApply = function(context, arg) {
if (typeof context === 'object') {
context = context || root;
} else {
context = Object.create(null);
} const fn = Symbol();
context[fn] = this;
context[fn](...arg);
delete context[fn];
} let person = {
name: 'Wango',
fn: function() {
console.log(this.name);
}
} function sayHi(age, sex) {
console.log(this.name, age, sex);
} sayHi.myApply(person, [24, 'male']);
// Wango 24 male

bind方法

// bind 改变this指向但是不立即执行函数,而是返回一个绑定了this的函数

// bind方法在绑定this指向的同时也可以传递参数
Function.prototype.myBind = function(context, ...innerArgs) {
// 不需要对参数类型进行判断,后边调用call方法时会进行处理
const fn = this;
// 不执行函数,而是返回一个函数等待调用
return function(...finalArgs) {
// 通过已经实现的call方法实现对this指向的改变,并传入参数执行
return fn.call(context, ...innerArgs, ...finalArgs);
}
} const person = {
name: 'Wango'
} function sayHi(age, sex) {
console.log(this.name, age, sex);
} const personSayHi_1 = sayHi.myBind(person, 24, 'male');
personSayHi_1();
// Wango 24 male
const personSayHi_2 = sayHi.myBind(person);
personSayHi_2(24, 'male');
// Wango 24 male

bind方法的核心在于在返回的函数中调用call方法

参考网址:call、apply和bind的实现

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

  1. JS核心系列:浅谈 call apply 与 bind

    在JavaScript 中,call.apply 和 bind 是 Function 对象自带的三个方法,这三个方法的主要作用是改变函数中的 this 指向,从而可以达到`接花移木`的效果.本文将对这 ...

  2. Javascript中call,apply,bind方法的详解与总结

    在 javascript之 this 关键字详解 文章中,谈及了如下内容,做一个简单的回顾: 1.this对象的涵义就是指向当前对象中的属性和方法. 2.this指向的可变性.当在全局作用域时,thi ...

  3. JS中call、apply、bind使用指南,带部分原理。

    为什么需要这些?主要是因为this,来看看this干的好事. box.onclick = function(){ function fn(){ alert(this); } fn();}; 我们原本以 ...

  4. Javascript中call、apply、bind函数

    javascript在函数创建的时候除了自己定义的参数外还会自动新增this和arguments两个参数 javascript中函数也是对象,call.apply.bind函数就是函数中的三个函数,这 ...

  5. 图解call、apply、bind的异同及各种实战应用演示

    一.图解call.apply.bind的异同 JavaScript中函数可以通过3种方法改变自己的this指向,它们是call.apply.bind.它们3个非常相似,但是也有区别.下面表格可以很直观 ...

  6. js里function的apply vs. bind vs. call

    js里除了直接调用obj.func()之外,还提供了另外3种调用方式:apply.bind.call,都在function的原型里.这3种方法的异同在stackoverflow的这个答案里说的最清楚, ...

  7. JS之apply,call,bind区别

    为了加深对基础知识的理解,今天再复习下js中的apply,call,bind的区别和用法.整理笔记的过程也是一个再次学习的过程. apply和call js中的调用apply和call方法可以改变某个 ...

  8. JavaScript中call、apply、bind、slice的使用

    1.参考资料 http://www.cnblogs.com/coco1s/p/4833199.html   2.归结如下 apply . call .bind 三者都是用来改变函数的this对象的指向 ...

  9. JS中的控制函数调用:call(),apply()和bind()

    所有的函数都具有call(),apply()和bind()方法.它们可以在执行方法的时候用一个值指向this,并改变面向对象的作用域. apply方法: 以下的两种表达式是等价的: func(arg1 ...

  10. JS中的call、apply、bind方法

    JS中的call.apply.bind方法 一.call()和apply()方法 1.方法定义 call方法: 语法:call([thisObj[,arg1[, arg2[,   [,.argN]]] ...

随机推荐

  1. 数理统计4:均匀分布的参数估计,次序统计量的分布,Beta分布

    接下来我们就对除了正态分布以外的常用参数分布族进行参数估计,具体对连续型分布有指数分布.均匀分布,对离散型分布有二项分布.泊松分布几何分布. 今天的主要内容是均匀分布的参数估计,内容比较简单,读者应尝 ...

  2. 在 .NET Core Logging中使用 Trace和TraceSource

    本文介绍了在.NET Core中如何在组件设计中使用Trace和TraceSource. 在以下方面会提供一些帮助: 1.你已经为.NET Framework和.NET Core / .NET Sta ...

  3. nginx反向代理signalr

    asp.net core应用常常要通过nginx来反向代理, 普通的web api配置asp.net core反向代理比较常见, 如果在应用中集成了signalr, 如何使用nginx来反代呢? ng ...

  4. G - 棋盘游戏

    小希和Gardon在玩一个游戏:对一个N*M的棋盘,在格子里放尽量多的一些国际象棋里面的"车",并且使得他们不能互相攻击,这当然很简单,但是Gardon限制了只有某些格子才可以放, ...

  5. 2019牛客暑期多校训练营(第四场)D-triples I

    >传送门< 题意:求最少需要多少个3的倍数按位或后可以得到数字a 思路:利用3的倍数对应的二进制数的性质来先选出一个x,然后根据数字a再配一个y出来 首先,我们都知道十进制中,任意一个数只 ...

  6. 【noi 2.6_7627】鸡蛋的硬度(DP)

    题意:其中n表示楼的高度,m表示你现在拥有的鸡蛋个数. 解法:f[i][j]表示 i 层楼有 j 个鸡蛋时,至少要扔多少次.3重循环,k为测试的楼层,分这时扔下去的鸡蛋碎和不碎的情况.要注意初始化. ...

  7. hdu 6794 Tokitsukaze and Multiple 前缀和思想+思维

    题意: t组输入,给你一个长度为n的数组,你每次可以从数组中找到a[i]和a[i+1],然后用a[i]+a[i+1]这个新元素来覆盖掉a[i]和a[i+1]的位置(1<=i<n),从而数组 ...

  8. Codeforces Round #645 (Div. 2) D. The Best Vacation (贪心,二分)

    题意:一年有\(n\)个月,每月有\(d_{i}\)天,找出连续的\(x\)天,使得这\(x\)天的日期总和最大,任意一年都能选. 题解:首先要先贪心,得到:连续的\(x\)天的最后一天一定是某个月的 ...

  9. 一张图解决ThreadLocal

    一张图解决ThreadLocal 一.前言 年底梳理知识体系时,研究了一下ThreadLocal的源码,整理了一张核心图. 想着,都走到这一步了,那就写一篇深度解读的文章吧.看过我之前文章的小伙伴都知 ...

  10. Python+Selenium+Mysql(动态获取数据,数据库交互)

    一.创建数据库连接 #!coding:utf-8 import pymysql ''' Python3之后不再支持MySQLdb的方式进行访问mysql数据库: 可以采用pymysql的方式 连接方式 ...