在议论bind()
要领之前我们先来看一道题目:
javascript
var altwrite = document.write; altwrite("hello"); //1.以上代码有什么题目 //2.准确操纵是如何的 //3.bind()要领怎样完成
关于上面这道题目,答案并非太难,重要考点就是this指向的题目,altwrite()函数转变this的指向global或window对象,致使执行时提醒不法挪用非常,准确的计划就是运用bind()
要领:
javascript
altwrite.bind(document)("hello")
固然也可以运用call()
要领:
javascript
altwrite.call(document, "hello")
本文的重点在于议论第三个题目bind()
要领的完成,在最先议论bind()
的完成之前,我们先来看看bind()
要领的运用:
绑定函数
bind()
最简朴的用法是建立一个函数,使这个函数不论怎样挪用都有一样的this值。罕见的毛病就像上面的例子一样,将要领从对象中拿出来,然后挪用,而且愿望this指向本来的对象。假如不做特别处置惩罚,平常会丧失本来的对象。运用bind()
要领可以很漂亮的处理这个题目:
javascript
this.num = 9; var mymodule = { num: 81, getNum: function() { return this.num; } }; module.getNum(); // 81 var getNum = module.getNum; getNum(); // 9, 由于在这个例子中,"this"指向全局对象 // 建立一个'this'绑定到module的函数 var boundGetNum = getNum.bind(module); boundGetNum(); // 81
偏函数(Partial Functions)
Partial Functions也叫Partial Applications,这里截取一段关于偏函数的定义:
Partial application can be described as taking a function that accepts some number of arguments, binding values to one or more of those arguments, and returning a new function that only accepts the remaining, un-bound arguments.
这是一个很好的特征,运用bind()
我们设定函数的预定义参数,然后挪用的时刻传入其他参数即可:
javascript
function list() { return Array.prototype.slice.call(arguments); } var list1 = list(1, 2, 3); // [1, 2, 3] // 预定义参数37 var leadingThirtysevenList = list.bind(undefined, 37); var list2 = leadingThirtysevenList(); // [37] var list3 = leadingThirtysevenList(1, 2, 3); // [37, 1, 2, 3]
和setTimeout一同运用
平常状况下setTimeout()
的this指向window或global对象。当运用类的要领时须要this指向类实例,就可以运用bind()
将this绑定到回调函数来治理实例。
javascript
function Bloomer() { this.petalCount = Math.ceil(Math.random() * 12) + 1; } // 1秒后挪用declare函数 Bloomer.prototype.bloom = function() { window.setTimeout(this.declare.bind(this), 1000); }; Bloomer.prototype.declare = function() { console.log('我有 ' + this.petalCount + ' 朵花瓣!'); };
注重:关于事宜处置惩罚函数和setInterval要领也可以运用上面的要领
绑定函数作为组织函数
绑定函数也适用于运用new操纵符来组织目的函数的实例。当运用绑定函数来组织实例,注重:this会被疏忽,然则传入的参数依然可用。
javascript
function Point(x, y) { this.x = x; this.y = y; } Point.prototype.toString = function() { return this.x + ',' + this.y; }; var p = new Point(1, 2); p.toString(); // '1,2' var emptyObj = {}; var YAxisPoint = Point.bind(emptyObj, 0/*x*/); // 完成中的例子不支撑, // 原生bind支撑: var YAxisPoint = Point.bind(null, 0/*x*/); var axisPoint = new YAxisPoint(5); axisPoint.toString(); // '0,5' axisPoint instanceof Point; // true axisPoint instanceof YAxisPoint; // true new Point(17, 42) instanceof YAxisPoint; // true
上面例子中Point和YAxisPoint同享原型,因而运用instanceof运算符推断时为true。
捷径
bind()
也可认为须要特定this值的函数制造捷径。
比方要将一个类数组对象转换为真正的数组,能够的例子以下:
javascript
var slice = Array.prototype.slice; // ... slice.call(arguments);
假如运用bind()
的话,状况变得更简朴:
javascript
var unboundSlice = Array.prototype.slice; var slice = Function.prototype.call.bind(unboundSlice); // ... slice(arguments);
完成
上面的几个小节可以看出bind()
有许多的运用场景,然则bind()
函数是在 ECMA-262 第五版才被到场;它能够没法在所有浏览器上运转。这就须要我们本身完成bind()
函数了。
起首我们可以经由过程给目的函数指定作用域来简朴完成bind()
要领:
javascript
Function.prototype.bind = function(context){ self = this; //保留this,即挪用bind要领的目的函数 return function(){ return self.apply(context,arguments); }; };
考虑到函数柯里化的状况,我们可以构建一个越发硬朗的bind()
:
javascript
Function.prototype.bind = function(context){ var args = Array.prototype.slice.call(arguments, 1), self = this; return function(){ var innerArgs = Array.prototype.slice.call(arguments); var finalArgs = args.concat(innerArgs); return self.apply(context,finalArgs); }; };
此次的bind()
要领可以绑定对象,也支撑在绑定的时刻传参。
继承,Javascript的函数还可以作为组织函数,那末绑定后的函数用这类体式格局挪用时,状况就比较玄妙了,须要涉及到原型链的通报:
javascript
Function.prototype.bind = function(context){ var args = Array.prototype.slice(arguments, 1), F = function(){}, self = this, bound = function(){ var innerArgs = Array.prototype.slice.call(arguments); var finalArgs = args.concat(innerArgs); return self.apply((this instanceof F ? this : context), finalArgs); }; F.prototype = self.prototype; bound.prototype = new F(); retrun bound; };
这是《JavaScript Web Application》一书中对bind()
的完成:经由过程设置一个中转组织函数F,使绑定后的函数与挪用bind()
的函数处于统一原型链上,用new操纵符挪用绑定后的函数,返回的对象也能一般运用instanceof,因而这是最严谨的bind()
完成。
关于为了在浏览器中能支撑bind()
函数,只须要对上述函数轻微修正即可:
javascript
Function.prototype.bind = function (oThis) { if (typeof this !== "function") { throw new TypeError("Function.prototype.bind - what is trying to be bound is not callable"); } var aArgs = Array.prototype.slice.call(arguments, 1), fToBind = this, fNOP = function () {}, fBound = function () { return fToBind.apply( this instanceof fNOP && oThis ? this : oThis || window, aArgs.concat(Array.prototype.slice.call(arguments)) ); }; fNOP.prototype = this.prototype; fBound.prototype = new fNOP(); return fBound; };
欢迎光临小弟博客:Superlin’s Blog
我的博客原文:Javascript中bind()要领的运用与完成