当前位置:网站首页>JS手写call apply bind (详细)(面试)
JS手写call apply bind (详细)(面试)
2022-08-03 15:08:00 【高高不想学习】
作用
这三方法都是编写js代码时常用的方法,目的是改变函数 this 的指向。
区别
执行时间:
call 和 apply 调用时直接执行,bind返回一个函数,何时调用何时执行。
传参方式:
apply 可以传一个类数组对象或者数组(只接收两个参数)
call 和 bind一个一个的接收参数
bind还有特殊的传参方式和调用方式(详细见下)
改变this方式:
call 和 apply 只是简单的改变当前调用函数时候的this指向
bind是返回一个新函数,这个函数的this指向永远执行绑定时候的那个对象
obj={};
fun.call(obj,1,2,3,4,5);
fun.apply(obj,[1,2,3,4,5]);
let otherFun=fun.bind(obj,1,2,3,4);//otherFun的this永远指向obj
执行时机 | 传参方式 | 改变this方式 | |
---|---|---|---|
call | 直接执行 | 一个一个接收 | 短暂改变 |
apply | 直接执行 | 类数组/数组 | 短暂改变 |
bind | 何时调用何时执行 | 一个一个接收 | 返回新函数,永久改变 |
代码:
call:在obj上加一个独一无二的属性(fn)表示调用call的那个函数,然后用obj调用这个函数(函数this指向调用者)这样就实现了this的转移。
arguments是形参的类数组形式,fun.call(obj,1,2,3,4) 调用call时arguments表示obj,1,2,3,4组成的类数组,可以通过数组方法splice(1)取后面的值,即1,2,3,4
function _call(obj) {
let fn = Symbol();
obj[fn] = this;
let arg = [...arguments].splice(1);
let result = obj[fn](arg);
delete obj[fn];
return result;
}
apply:跟call大致一样,只不过传参方式不一样
function _apply(obj) {
let fn = Symbol();
obj[fn] = this;
let result = obj[fn](...arguments[1]);
delete obj[fn];
return result;
}
bind:它不用用obj来保存函数,因为它本身返回一个新函数。(不是最终版)
function bind(obj) {
let content = this;
let arr = Array.prototype.slice.call(arguments, 1);
return function () {
return content.apply(obj, arr);
}
}
ES6将类数组形式转换为数组形式的两者种方法:
Array.prototype.slice.call(arguments)
[].slice.call(arguments)
bind特性:
1. 传参方式 bind(obj,1,2,3,4)(5)这是返回新函数之后直接调用
这就需要考虑第二个括号的arguments的值了
function _bind(obj) {
let content = this;
let arr = Array.prototype.slice.call(arguments, 1);
return function () {
let new_arr = Array.prototype.slice.call(arguments);
return content.apply(obj, arr.concat(new_arr));
}
}
用数组方法 concat() 来连接第一个arguments的参数和第二个arguments的参数
2. bind返回的函数可以被new 此时传入的obj应改该被无视掉。构造函数的this应该永远指向新创建的实例对象
function _bind(obj) {
let content = this;
let arr = Array.prototype.slice.call(arguments, 1);
let finalBind = function () {
if (this instanceof finalBind) {//为true时,表示finalBind被当成构造函数使用。为false,表示正常使用
obj = this;//忽略对原函数this的操作
}
let new_arr = Array.prototype.slice.call(arguments);
return content.apply(obj, arr.concat(new_arr));
}
return finalBind
}
总结
call , apply , bind 都说改变this执行的函数,每个函数都可以调用。注意三者的区别
bind会被当成构造函数是一个难点,此时需要取消对原函数this指向的操作。
边栏推荐
猜你喜欢
面试官都震惊,你这“网络基础”可以啊
如何把MapGIS的区文件转为ArcGIS的SHAPE面文件
问题7:功能测试花瓶用例
南京一研究所回应招聘硕士保安:负责安全生产等,48人选1
R7 6800H+RTX3050+120Hz 2.8K OLED screen, Intrepid Pro15 2022 pre-sale
2021年12月电子学会图形化三级编程题解析含答案:分身术
How to use redis
The general trend, another key industry related to Sino-US competition, has reached a critical moment
WMS软件国内主要供应商分析
PostgreSQL V14中更好的SQL函数
随机推荐
使用华为HECS云服务器打造Telegraf+Influxdb+Grafana 监控系统【华为云至简致远】
2022-随便学学
Linux安装Mysql的几种方法
高性能创作本,日常修图剪辑选华硕无畏Pro15 2022完全足矣!
2022年镇海夏令营组合数学和数论班 —— 数学作业 1
Jupyter Notebook 交互式编程 & 低代码拖拽式编程 | 数据科学生态下的理想平台
HDU 1160 FatMouse's Speed(最长递减子序列变形)
2021年12月电子学会图形化四级编程题解析含答案:新冠疫苗接种系统
【报错】ImportError: libpng16-57e5e0a0.so.16.37.0: cannot open shared object file: No such file or direc
2021年12月电子学会图形化二级编程题解析含答案:消灭蝙蝠
2021年12月电子学会图形化四级编程题解析含答案:聪明的小猫
Controller层代码这么写,简洁又优雅!
2021年12月电子学会图形化四级编程题解析含答案:森林运动会
一个在浏览器中看到的透视Cell实现
R7 6800H+RTX3050+120Hz 2.8K OLED屏,无畏Pro15 2022开启预售
使用虚幻引擎自动化工具实现自动化部署
liunx服务器遇到SYN_SENT洪水攻击
eolink告诉你,国内Api行业,可以内卷到什么程度?
DeepLink在转转的实践
devops-2:Jenkins的使用及Pipeline语法讲解