当前位置:网站首页>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指向的操作。
边栏推荐
- PAT乙级-B1009 说反话(20)
- ffplay视频播放原理分析
- 三元表达式实现多个条件的判断
- 你没见过的《老友记》镜头,AI给补出来了|ECCV 2022
- 分布式系统与微服务的区别
- 【R语言科研绘图】--- 柱状图
- Use Typora+EasyBlogImageForTypora to write a blog and upload pictures quickly without a picture bed
- 2021年12月电子学会图形化四级编程题解析含答案:质数判断器
- 手摸手带你完成智慧路灯构建及避坑【华为云至简致远】
- Jupyter Notebook 交互式编程 & 低代码拖拽式编程 | 数据科学生态下的理想平台
猜你喜欢

又有大厂员工连续加班倒下/ 百度搜狗取消快照/ 马斯克生父不为他骄傲...今日更多新鲜事在此...

面试官都震惊,你这“网络基础”可以啊

Controller层代码这么写,简洁又优雅!

彻底搞懂云桌面配置及实践踩坑【华为云至简致远】

The general trend, another key industry related to Sino-US competition, has reached a critical moment

今日睡眠质量记录75分

eolink告诉你,国内Api行业,可以内卷到什么程度?

How to use redis

Huffman tree

2021年12月电子学会图形化四级编程题解析含答案:棕熊大战
随机推荐
【软件工程之美 - 专栏笔记】36 | DevOps工程师到底要做什么事情?
动态链接库.dll、.so和静态库.a,cmake指令
Flink作业调度详解
【周报】2022年7月31日
彻底搞懂云桌面配置及实践踩坑【华为云至简致远】
【FPGA教程案例44】图像案例4——基于FPGA的图像中值滤波verilog实现,通过MATLAB进行辅助验证
【重构map】【重构filter】【重构Some】【重构reduce方法】【重构flat函数】
程序员面试必备PHP基础面试题 – 第二十天
问题5:发现缺陷怎么办?缺陷的类型有哪些?
使用Typora+EasyBlogImageForTypora写博客,无图床快速上传图片
LeetCode169:多数元素
【网络结构】VGG
PAT乙级-B1014 福尔摩斯的约会(20)
ubiquant量化竞赛
[The Beauty of Software Engineering - Column Notes] 36 | What exactly do DevOps engineers do?
【问题】torch和torchvision对应版本
【报错】import cv2 as cv ModuleNotFoundError: No module named ‘cv2
程序员面试必备PHP基础面试题 – 第二十一天
MATLAB中writetimetable函数用法
理解string类