在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的更多相关文章

  1. 箭头函数表达式和声名式函数表达式的区别以及 Function.prototype的bind, apply,call方法

    箭头函数不能用做构造函数 箭头函数没有arguments参数 箭头函数没有自己的this,是从作用域链上取this,是与箭头函数定义的位置有关的,与执行时谁调用无关,所以用call,apply,bin ...

  2. Function.prototype.bind接口浅析

    本文大部分内容翻译自 MDN内容, 翻译内容经过自己的理解. https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Glo ...

  3. 一起Polyfill系列:Function.prototype.bind的四个阶段

    昨天边参考es5-shim边自己实现Function.prototype.bind,发现有不少以前忽视了的地方,这里就作为一个小总结吧. 一.Function.prototype.bind的作用 其实 ...

  4. JavaScript 函数绑定 Function.prototype.bind

    ECMAScript Edition5 IE9+支持原生,作用为将一个对象的方法绑定到另一个对象上执行. Function.prototype.bind = Function.prototype.bi ...

  5. Function.prototype.bind

    解析Function.prototype.bind 简介 对于一个给定的函数,创造一个绑定对象的新函数,这个函数和之前的函数功能一样,this值是它的第一个参数,其它参数,作为新的函数的给定参数. b ...

  6. 解析Function.prototype.bind

    简介 对于一个给定的函数,创造一个绑定对象的新函数,这个函数和之前的函数功能一样,this值是它的第一个参数,其它参数,作为新的函数的给定参数. bind的作用 bind最直接的作用就是改变this的 ...

  7. javascript Function.prototype.bind

    语法: fn.bind(obj,arg1,arg2,arg3...) bind是es5新增的方法,顾名思义,它的作用是将函数绑定到某个对象上,就像是某个对象调用方法一样.其本质还是改变了该函数的上下文 ...

  8. 理解javascript中的Function.prototype.bind

    在初学Javascript时,我们也许不需要担心函数绑定的问题,但是当我们需要在另一个函数中保持上下文对象this时,就会遇到相应的问题了,我见过很多人处理这种问题都是先将this赋值给一个变量(比如 ...

  9. 浅析 JavaScript 中的 Function.prototype.bind() 方法

    Function.prototype.bind()方法 bind() 方法的主要作用就是将函数绑定至某个对象,bind() 方法会创建一个函数,函数体内this对象的值会被绑定到传入bind() 函数 ...

随机推荐

  1. HTML连载20-并集选择器&兄弟选择器

    一.并集选择器 1.作用:给所有的选择器选中的标签设置属性. 2.格式: 选择器1,选择器2{ 属性:值: } 3.例如: .abc1,#abc2{ color:red; } .......省略代码. ...

  2. 怎样用 C# 快速比较 2 个文件是否是相同的文件?

    方案1: 直接贴代码了: using System; using System.Collections.Generic; using System.IO; using System.Linq; usi ...

  3. kali渗透综合靶机(十八)--FourAndSix2靶机

    kali渗透综合靶机(十八)--FourAndSix2靶机 靶机下载地址:https://download.vulnhub.com/fourandsix/FourAndSix2.ova 一.主机发现 ...

  4. ASP.Net Core使用Ajax局部更新

    由于目前ASP.NET Core中没有提供Ajax帮助器,所以参照 上一篇帖文,使用data-ajax-*属性来使用jQuery Unobtrusive Ajax功能实现HTML的局部页面元素更新. ...

  5. Spring Boot @EnableAutoConfiguration解析

    刚做后端开发的时候,最早接触的是基础的spring,为了引用二方包提供bean,还需要在xml中增加对应的包<context:component-scan base-package=" ...

  6. Vue.js项目实战-多语种网站(租车)

    首先来看一下网站效果,想写这个项目的读者可以自行下载哦,地址:https://github.com/Stray-Kite/Car: 在这个项目中,我们主要是为了学习语种切换,也就是右上角的 中文/En ...

  7. i春秋四周年庆典狂欢丨价值6000元的Web安全课程免费送啦

    重磅好消息 i春秋四周年庆典狂欢 感恩回馈新老用户 5888元的Web安全线上提高班 988元的Web安全线上入门班 免费送啦 快来围观 活动详情 1.活动时间:6月17日—6月30日 2.活动规则: ...

  8. dlopen动态链接库操作

    void *dlopen(const char *filename, int flag); //打开一个动态链接库,并返回动态链接库的句柄 char *dlerror(void); void *dls ...

  9. ca动画

    //动画上下文-(void)animationOfUIKit{    UIView *redView=[[UIView alloc]initWithFrame:CGRectMake(10, 10, 1 ...

  10. K-means 和 EM 比较

    回顾 前几篇对 k-means 有过理解和写了一版伪代码, 因为思想比较非常朴素, 就是初始化几个中心点, 然后通过计算距离的方式, "物以类聚", 不断迭代中心点, 最后收敛, ...