Function.prototype.call.bind
在JavaScript中借用方法
在JavaScript中,有时候需要在一个不同的对象上重用一个函数,而不是在定义它的对象或者原型中。通过使用call(),applay()和bind(),我们可以很方便地从不同的对象借用方法,而不需要继承它们 – 这是一个在专业JavaScript开发者的工具箱中很有用的工具。
这篇文章假设你已经充分了解了call(),apply() 和 bind() 以及它们的不同点。
在JavaScript中,你接触的几乎所有东西都是一个对象,除了string,number 和 booleans这样不可变的原始值。一个数组是一种对象类型,适合用于有序数据列表的遍历和修改,在它的原型中有很多有用的方法,例如slice,join,push 和 pop。
我们看到对象最常见的使用情况就是从一个数组中借用方法,因为它们都是列表类型的数据结构。最常被借用的方法是 Array.prototype.slice。
function myFunc() {
// 错误, arguments是一个类数组对象, 不是一个真实的数组
arguments.sort();
// 借用 Array 原型中的方法 slice, 它接受一个类数组对象 (key:value)
// 并返回一个真实的数组
var args = Array.prototype.slice.call(arguments);
// args 现在是一个真正的数组, 所以可以使用Array的sort()方法
args.sort();
}
myFunc('bananas', 'cherries', 'apples');
借用方法是可行的,因为call和apply允许我们在一个不同的上下文中调用方法,是一个很好的方式来重用已经存在的函数,而无需让一个对象扩展自另一个对象。数组实际上在它的原型中定义了很多方法,而且一般是可重用的,下面再举两个例子是join和filter:
// 接收一个字符串 "abc" 并输出 "a|b|c
Array.prototype.join.call('abc', '|');
// 接收一个字符串并移除所有的非元音字母
Array.prototype.filter.call('abcdefghijk', function(val) {
return ['a', 'e', 'i', 'o', 'u'].indexOf(val) !== -;
}).join('');
正如你所看到的,不仅仅对象可以得益于从数组中借用方法,字符串也可以。然而,因为方法一般被定义在了原型上,所以每次我们借用方法都要写String.prototype 或者 Array.prototype,这是很多余并且很烦人的事。一种有效的方法的是使用Literals(字面量)。
Literal是一种语法语言结构, 遵循 JavaScript 的规则,MDN 上这样解释:
你可以使用literals来表示JavaScript中的值。这些都是固定的值,不是变量,你可以再脚本中按照字面写出来。
Literals允许我们以缩略的形式访问原型方法:
[].slice.call(arguments);
[].join.call('abc', '|');
''.toUpperCase.call(['lowercase', 'words', 'in', 'a', 'sentence']).split(',');
这样变得简单些了,但看起来还是有点丑,还是要使用 [] 和 ""来借用它们的方法。我们可以更进一步缩短,通过存储一个引用,把字面量和它的方法作为一个变量:
var slice = [].slice;
slice.call(arguments); var join = [].join;
join.call('abc', '|'); var toUpperCase = ''.toUpperCase;
toUpperCase.call(['lowercase', 'words', 'in', 'a', 'sentence']).split(',');
通过引用被借用的方法,我们可以很方便地使用call()调用它,享受所有可重用性的好处。为了继续减少冗长,我们来看一下是否可以在每次调用的时候,不写call() 或者 apply() 就能借用一个方法:
var slice = Function.prototype.call.bind(Array.prototype.slice);
slice(arguments); var join = Function.prototype.call.bind(Array.prototype.join);
join('abc', '|'); var toUpperCase = Function.prototype.call.bind(String.prototype.toUpperCase);
toUpperCase(['lowercase', 'words', 'in', 'a', 'sentence']).split(',');
如你所见,通过使用Function.prototype.call.bind,我们现在可以静态绑定“被借用”的来自不同本地原型的方法,但是var slice = Function.prototype.call.bind(Array.prototype.slice)到底是如何工作的呢?
Function.prototype.call.bind 乍一眼看起来有些复杂,但是理解它是如何工作的非常有用。
Function.prototype.call是一个引用,用来调用一个函数并且把它的“this”值设置为使用内部提到的方法。记住“bind”返回一个新的函数,这个函数总是会牢记它的“this”值。因此,
.bind(Array.prototype.slice)会返回一个新的函数,它的“this”被永久地设置成了Array.prototype.slice函数。
通过结合以上两个,我们现在有了一个新的函数,它将会调用“call”函数并且“this”限定为了“slice”函数。简单地调用slice()便可以引用之前限定的方法。
继承很棒,但是当程序员想要重用一些对象或者模块中的常见功能时会经常求助于它。如果你正在单独使用继承来重用代码,你可能会做错事,在很多情况下,简单的借用一个方法会变得非常麻烦。
到目前为止,我们仅仅讨论了借用本地方法,但其实可以借用任何方法!使用下面的代码来计算一个运动员所得的比赛分数:
var scoreCalculator = {
getSum: function(results) {
var score = ;
for (var i = , len = results.length; i < len; i++) {
score = score + results[i];
}
return score;
},
getScore: function() {
return scoreCalculator.getSum(this.results) / this.handicap;
}
};
var player1 = {
results: [, , ],
handicap:
};
var player2 = {
results: [, , ],
handicap:
};
var score = Function.prototype.call.bind(scoreCalculator.getScore);
// Score: 24.375
console.log('Score: ' + score(player1));
// Score: 17
console.log('Score: ' + score(player2));
尽管上面的例子是人为的,但是很容易看到用户定义的方法也能像本地方法那样被方便地借用。
Call, bind 和 apply 允许我们改变函数被调用的方式,在借用一个函数时经常被使用。大多数开发者都很熟悉借用本地方法,但是很少知道用户定义的方法也可以。
在过去的几年里,JavaScript中的函数式编程逐渐增多,我希望简短的使用Function.prototype.call.bind来借用方法将变得越来越普遍。
链接:https://www.zcfy.cc/article/borrowing-methods-in-javascript-by-david-shariff-794.html
Function.prototype.call.bind的更多相关文章
- 箭头函数表达式和声名式函数表达式的区别以及 Function.prototype的bind, apply,call方法
箭头函数不能用做构造函数 箭头函数没有arguments参数 箭头函数没有自己的this,是从作用域链上取this,是与箭头函数定义的位置有关的,与执行时谁调用无关,所以用call,apply,bin ...
- Function.prototype.bind接口浅析
本文大部分内容翻译自 MDN内容, 翻译内容经过自己的理解. https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Glo ...
- 一起Polyfill系列:Function.prototype.bind的四个阶段
昨天边参考es5-shim边自己实现Function.prototype.bind,发现有不少以前忽视了的地方,这里就作为一个小总结吧. 一.Function.prototype.bind的作用 其实 ...
- JavaScript 函数绑定 Function.prototype.bind
ECMAScript Edition5 IE9+支持原生,作用为将一个对象的方法绑定到另一个对象上执行. Function.prototype.bind = Function.prototype.bi ...
- Function.prototype.bind
解析Function.prototype.bind 简介 对于一个给定的函数,创造一个绑定对象的新函数,这个函数和之前的函数功能一样,this值是它的第一个参数,其它参数,作为新的函数的给定参数. b ...
- 解析Function.prototype.bind
简介 对于一个给定的函数,创造一个绑定对象的新函数,这个函数和之前的函数功能一样,this值是它的第一个参数,其它参数,作为新的函数的给定参数. bind的作用 bind最直接的作用就是改变this的 ...
- javascript Function.prototype.bind
语法: fn.bind(obj,arg1,arg2,arg3...) bind是es5新增的方法,顾名思义,它的作用是将函数绑定到某个对象上,就像是某个对象调用方法一样.其本质还是改变了该函数的上下文 ...
- 理解javascript中的Function.prototype.bind
在初学Javascript时,我们也许不需要担心函数绑定的问题,但是当我们需要在另一个函数中保持上下文对象this时,就会遇到相应的问题了,我见过很多人处理这种问题都是先将this赋值给一个变量(比如 ...
- 浅析 JavaScript 中的 Function.prototype.bind() 方法
Function.prototype.bind()方法 bind() 方法的主要作用就是将函数绑定至某个对象,bind() 方法会创建一个函数,函数体内this对象的值会被绑定到传入bind() 函数 ...
随机推荐
- MySQL数据库中查询表的所有列名
MySQL数据库中: 查询某个数据库中某个表的所有列名 SELECT COLUMN_NAME FROM information_schema.COLUMNS WHERE TABLE_SCHEMA = ...
- switch的对象不能为null
我写的NPE 虽然不多, 但几乎每次系统出问题的时候,看到api返回值是空的,绝大多数是NPE造成的. 这时候会感慨一下谁写的bug,然后去补判空代码. 最近抽风,开始给自己写的代码添加UnitTes ...
- vue的双向绑定原理浅析与简单实现
很久之前看过vue的一些原理,对其中的双向绑定原理也有一定程度上的了解,只是最近才在项目上使用vue,这才决定好好了解下vue的实现原理,因此这里对vue的双向绑定原理进行浅析,并做一个简单的实现. ...
- .net core EF Core 调用存储过程
在这里,我们将尝试去学习一下 .net core EF Core 中调用存储过程. 我们知道,EF Core 是不支持直接调用存储过程的,那它又提供了什么样的方式去执行存储过程呢?有如下方法: 1.F ...
- 初探云原生应用管理之:聊聊 Tekton 项目
[编者的话]“人间四月芳菲尽,山寺桃花始盛开.” 越来越多专门给 Kubernetes 做应用发布的工具开始缤纷呈现,帮助大家管理和发布不断增多的 Kubernetes 应用.在做技术选型的时候,我们 ...
- Yapi接口管理平台 本地部署 windows环境 -
YApi 是高效.易用.功能强大的 api 管理平台,旨在为开发.产品.测试人员提供更优雅的接口管理服务.可以帮助开发者轻松创建.发布.维护 API,YApi 还为用户提供了优秀的交互体验,开发人员只 ...
- jmeter返回结果出现乱码
这是我的问题: 请求的百度地址,返回的数据是中文乱码 去jmeter安装目录的bin下: 修改配置文件jmeter.properties 右键打开 ,查找:sampleresult.default.e ...
- HTML input属性详谈
value属性 value属性指定输入字段的初始值: <form> 名字:<br> <input type="text" name="you ...
- linux线程绑定cpu
函数介绍 #define __USE_GNU #include <sched.h> void CPU_ZERO(cpu_set_t *set); void CPU_SET(int cpu, ...
- rhel安装输入法
# yum install "@Chinese Support" 安装完成后,设置输入法: System -> Preferences -> Input Method