JS中bind、call和apply的作用以及在TS装饰器中的用法
1,前言
bind
、call
和apply
在函数式编程时候非常有用,本文旨在记录一下我遇到过的一些用法和知识点,也记录一下在装饰器中的用法。
1,call
call()
方法使用一个指定的this
值和单独给出的一个或多个参数来调用一个函数。它的第一个参数是你需要指向的this
目标,后面的参数是你需要传递的参数,无参数可以不写。
语法:
function.call(target, arg1, arg2, ...)
1.1,例子
如下,控制台会打印出:快看【张三】在奔跑
const Person = {
Name: '张三',
Run() {
return `快看【${this.Name}】在奔跑`
}
}
const Animal = {
Name: '猛犸象'
}
console.log(Person.Run()) // 打印出:快看【张三】在奔跑
让我们使用call
改变下this
指向
1.2,直接调用
如果没有传递第一个参数,this
的值将会被绑定为全局对象,也就是window
对象(浏览器环境)。由于在window
上找不到this.Name
这个属性,控制台会打印出:快看【undefined】在奔跑
console.log(Person.Run.call()) // 打印出:快看【undefined】在奔跑
1.3,将this指向另一个对象
此时this
会绑定为被指向的对象,控制台会打印出:快看【猛犸象】在奔跑
console.log(Person.Run.call(Animal)) // 打印出:快看【猛犸象】在奔跑
1.4,传递参数
const Person = {
Name: '张三',
Run(param1, param2) {
console.log(param1)
console.log(param2)
return `快看【${this.Name}】在奔跑`
}
}
const Animal = {
Name: '猛犸象'
}
console.log(Person.Run.call(Animal, 10, '100')) // 打印出:10、'100'、快看【猛犸象】在奔跑
2,apply
apply()
方法调用一个具有给定this
值的函数,以及以一个数组(或类数组对象)的形式提供的参数。它的第一个参数是你需要指向的this
目标,后面的参数是你需要传递的数组参数,无参数可以不写。
语法:
function.apply(target, [argsArray])
2.1,例子
如下,控制台会打印出:快看【张三】在奔跑
const Person = {
Name: '张三',
Run() {
return `快看【${this.Name}】在奔跑`
}
}
const Animal = {
Name: '猛犸象'
}
console.log(Person.Run()) // 打印出:快看【张三】在奔跑
让我们使用apply
改变下this
指向
2.2,直接调用
如果没有传递第一个参数,this
的值将会被绑定为全局对象,也就是window
对象(浏览器环境)。由于在window
上找不到this.Name
这个属性,控制台会打印出:快看【undefined】在奔跑
console.log(Person.Run.apply()) // 打印出:快看【undefined】在奔跑
2.3,将this指向另一个对象
此时this
会绑定为被指向的对象,控制台会打印出:快看【猛犸象】在奔跑
console.log(Person.Run.apply(Animal)) // 打印出:快看【猛犸象】在奔跑
2.4,传递参数
const Person = {
Name: '张三',
Run(...arg) {
console.log(arg)
return `快看【${this.Name}】在奔跑`
}
}
const Animal = {
Name: '猛犸象'
}
console.log(Person.Run.apply(Animal, [10, '100'])) // 打印出:[10、'100']、快看【猛犸象】在奔跑
2.5,合并数组
let arr = ['a', 'b']
let elements = [0, 1, 2]
array.push.apply(arr, elements)
console.log(arr) // ["a", "b", 0, 1, 2]
3,bind
bind()
方法创建一个新的函数,在bind()
被调用时,这个新函数的this
被指定为bind()
的第一个参数,而其余参数将作为新函数的参数,供调用时使用。
语法:
function.bind(target, arg1, arg2, ...)
3.1,例子
如下,控制台会打印出:快看【张三】在奔跑
const Person = {
Name: '张三',
Run() {
return `快看【${this.Name}】在奔跑`
}
}
const Animal = {
Name: '猛犸象'
}
console.log(Person.Run()) // 打印出:快看【张三】在奔跑
让我们使用apply
改变下this
指向
3.2,直接调用
如果没有传递第一个参数,this
的值将会被绑定为全局对象,也就是window
对象(浏览器环境)。由于在window
上找不到this.Name
这个属性,控制台会打印出:快看【undefined】在奔跑
注意:bind
返回的是一个方法,需要加上()
执行才行
console.log(Person.Run.bind()()) // 打印出:快看【undefined】在奔跑
3.3,将this指向另一个对象
此时this
会绑定为被指向的对象,控制台会打印出:快看【猛犸象】在奔跑
console.log(Person.Run.bind(Animal)()) // 打印出:快看【猛犸象】在奔跑
3.4,传递参数
const Person = {
Name: '张三',
Run(param1, param2) {
console.log(param1)
console.log(param2)
return `快看【${this.Name}】在奔跑`
}
}
const Animal = {
Name: '猛犸象'
}
console.log(Person.Run.bind(Animal, 996, '100')()) // 打印出:996 '100' 快看【猛犸象】在奔跑
4,TypeScript中装饰器使用
使用bind
或者apply
或者call
,可以将方法装饰器中的this
指向被装饰的方法,不影响原方法使用的同时,注入新的逻辑处理。
function GetHttp(param: string) {
return function (target: any, Name: any, desc: any): void {
console.log(target) // 原型
console.log(Name) // 方法名
console.log(desc) // 方法描述 desc.value即是该方法
const ev = desc.value
desc.value = function(): void {
console.log('我是改写后的function')
ev.call(this)
}
}
}
class HttpGet {
name: string
constructor(name: string) {
this.name = name
}
@GetHttp('方法装饰器')
request(): void {
console.log(this.name)
}
}
const HttpObj = new HttpGet('小红')
HttpObj.request()
// 打印出:方法装饰器、我是改写后的function、小红
5,总结
5.1,相同点
- 都可以通过指定第一个参数,改变
this
指向 - 都可以传递参数
5.2,不同点
bind
返回的是一个函数,需要加上()
来执行apply
传递参数需要数组的形式
如果看了觉得有帮助的,我是@鹏多多,欢迎 点赞 关注 评论;END
PS:在本页按F12,在console中输入document.querySelectorAll('.diggit')[0].click(),有惊喜哦
公众号
往期文章
- 助你上手Vue3全家桶之Vue-Router4教程
- 助你上手Vue3全家桶之Vue3教程
- 助你上手Vue3全家桶之VueX4教程
- 使用nvm管理node.js版本以及更换npm淘宝镜像源
- 超详细!Vue-Router手把手教程
- vue中利用.env文件存储全局环境变量,以及配置vue启动和打包命令
- 微信小程序实现搜索关键词高亮
- 超详细!Vue的九种通信方式
- 超详细!Vuex手把手教程
个人主页
JS中bind、call和apply的作用以及在TS装饰器中的用法的更多相关文章
- python_如何修改装饰器中参数?
案例: 为分析程序内哪些函数执行时间开销较大,我们需定义一个带timeout参数的装饰器 需求: 统计被装饰函数的运行时间 时间大于timeout时,将此次函数调用记录到log日志中 运行时可以修改t ...
- 快速理解js中的call,apply的作用
今天被人问到js中的call,apply的区别和用途,解释了一番后,想到之前在逼乎上看到一位小伙伴生动形象的解释 本身不难理解,看下MDN就知道了,但是不常用,遇到了,还要脑回路回转下.或者时间长了, ...
- python装饰器中@wraps作用--修复被装饰后的函数名等属性的改变
Python装饰器(decorator)在实现的时候,被装饰后的函数其实已经是另外一个函数了(函数名等函数属性会发生改变),为了不影响,Python的functools包中提供了一个叫wraps的de ...
- 【转】odoo 新API装饰器中one、model、multi的区别
http://blog.csdn.net/qq_18863573/article/details/51114893 1.one装饰器详解 odoo新API中定义方式: date=fields.Date ...
- 装饰器中functools的用处
定义一个最简单的装饰器 def user_login_data(f): def wrapper(*args, **kwargs): return f(*args, **kwargs) return w ...
- 装饰器中的@functools.wraps的作用
def login_required(view_func): @functools.wraps(view_func) def wrapper(*args, **kwargs): ...... retu ...
- python装饰器中functools.wraps的作用详解
直接上代码看效果: # 定义一个最简单的装饰器 def user_login_data(f): def wrapper(*args, **kwargs): return f(*args, **kwar ...
- python装饰器中的计时器thd.strat用法
thd = KThread(target=_new_func, args=(), kwargs=new_kwargs) thd.start() thd.join(seconds) alive = th ...
- JSP和Servlet中的几个编码的作用及原理
首先,说说JSP和Servlet中的几个编码的作用. 在JSP和Servlet中主要有以下几个地方可以设置编码,pageEncoding="UTF-8".contentType=& ...
随机推荐
- Redis缓存穿透、缓存击穿、缓存雪崩的介绍及其解决方案
首先,来画一张图了解下缓存处理的流程 一.缓存穿透 描述: 缓存穿透是指缓存和数据库中都没有的数据,而用户不断发起请求查询该数据,导致数据库压力过大. 解决方案: 1.接口校验 如鉴权校验.数据合法性 ...
- go 连接MSSQLServer数据库【遇到的坑】
前言:项目测试需要用到mssqlserver数据库连接,遇到坑,自己爬直接上代码: package main import ( "database/sql" " ...
- ☕【Java技术指南】「编译器专题」深入分析探究“静态编译器”(JAVA\IDEA\ECJ编译器)是否可以实现代码优化?
技术分析 大家都知道Eclipse已经实现了自己的编译器,命名为 Eclipse编译器for Java (ECJ). ECJ 是 Eclipse Compiler for Java 的缩写,是 Jav ...
- 靶场渗透CH4INRULZ_v1.0.1
最新文章见我个人博客:点此 靶机环境下载地址:[下载] ova下载下来后直接导入virtualbox即可(https://www.vulnhub.com/entry/ch4inrulz-101,247 ...
- leetcode347 —— n中topK && PriorityQueue(Heap) && Map遍历
题目要求:求前K个最频繁出现的数字. 1.很容易想到,使用HashMap<Integer,Integer>来存储<number,frequency>键值对 1 int n = ...
- 【UE4 C++】学习笔记汇总
UE4 概念知识 基础概念--文件结构.类型.反射.编译.接口.垃圾回收.序列化[导图] GamePlay架构[导图] 类的继承层级关系[导图] 反射机制 垃圾回收机制/算法 序列化 Actor 的生 ...
- 【UE4 C++ 基础知识】<3> 基本数据类型、字符串处理及转换
基本数据类型 TCHAR TCHAR就是UE4通过对char和wchar_t的封装 char ANSI编码 wchar_t 宽字符的Unicode编码 使用 TEXT() 宏包裹作为字面值 TCHAR ...
- 改善深层神经网络-week1编程题(Initializaion)
Initialization 如何选择初始化方式,不同的初始化会导致不同的结果 好的初始化方式: 加速梯度下降的收敛(Speed up the convergence of gradient desc ...
- Sequence Model-week3编程题2-Trigger Word Detection
1. Trigger Word Detection 我们的触发词将是 "Activate.".每当它听到你说 "Activate.",它就会发出 "c ...
- 谈谈BEM规范(含代码)
css规范之BEM规范 前言 引用一句经典名言在编程的世界里其中一件最难的事情就是命名,不管是设计到编程语言还是标记语言都会有命名的需求.今天聊的就是关于css的命名规范的发展过程以及演变. 命名的发 ...