当前位置:网站首页>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指向的操作。
边栏推荐
- LeetCode15:三数之和
- 问题10:注册页面的易用性测试?
- 兆骑科创高层次人才引进平台,创新创业赛事活动路演
- 使用Typora+EasyBlogImageForTypora写博客,无图床快速上传图片
- 兔起鹘落全端涵盖,Go lang1.18入门精炼教程,由白丁入鸿儒,全平台(Sublime 4)Go lang开发环境搭建EP00
- leetcode-105 从前序与中序遍历序列构造二叉树-使用栈代替递归
- R7 6800H+RTX3050+120Hz 2.8K OLED屏,无畏Pro15 2022开启预售
- JS每晚24:00更新某方法
- 【R语言科研绘图】--- 柱状图
- MySQL性能优化的'4工具+10技巧'
猜你喜欢
随机推荐
程序员面试必备PHP基础面试题 – 第二十天
【指针内功修炼】函数指针 + 函数指针数组 + 回调函数(二)
【报错】import cv2 as cv ModuleNotFoundError: No module named ‘cv2
一次做数据报表的踩坑经历,让我领略了数据同步增量和全量的区别
redis的使用方法
地球自转加快
夜神浏览器fiddler抓包
web漏洞之远程命令/代码执行
Mysql 生成排序序号
PHP中高级面试题 – 第三天
今日睡眠质量记录75分
生物统计师与临床医生协同研究使用的低代码洞察平台丨数据科学 x 临床医学
WMS软件国内主要供应商分析
2021年12月电子学会图形化三级编程题解析含答案:数星星
ubiquant量化竞赛
gocron定时任务管理系统的安装与运行
Huffman tree
在北极都可以穿短袖了,温度飙升至32.5℃
选择合适的 DevOps 工具,从理解 DevOps 开始
问题3:你提交的缺陷开发认为这不是BUG,怎么办?









