前言
bind()接受无数个参数,第一个参数是它生成的新函数的this指向,比如我传个window,不管它在何处调用,这个新函数中的this就指向window,这个新函数的参数就是bind()的第二个、第三个、第四个....第n个参数加上它原本的参数。(行吧,我自己都蒙圈了)
示例介绍
我们还是看看栗子比较好理解,举个bind()最基本的使用方法:
this.x = 9;
var module = {
x: 81,
getx: function() { return this.x; }
};
module.getx(); // 返回 81
var retrievex = module.getx;
retrievex(); // 返回 9, 在这种情况下,"this"指向全局作用域
// 创建一个新函数,将"this"绑定到module对象
// 新手可能会被全局的x变量和module里的属性x所迷惑
var boundgetx = retrievex.bind(module);
boundgetx(); // 返回 81
这里很明显,我们在window对象下调用retrievex,得到的结果肯定是window下的x,我们把module对象绑定到retrievex的this上,问题就解决了,不管它在何处调用,this都是指向module对象。
还有bind()的其他参数,相信第一次接触bind()的朋友看到上面的定义都会蒙圈。
还是举个栗子:
function list() {
return array.prototype.slice.call(arguments);
}
var list1 = list(1, 2, 3); // [1, 2, 3]
// 创建一个拥有预设初始参数的函数
var leadingthirtysevenlist = list.bind(undefined,[69,37],{a:2});
var list2 = leadingthirtysevenlist(); // [[69,37],{a:2}]
var list3 = leadingthirtysevenlist(1, 2, 3); // [[69,37],{a:2}, 1, 2, 3]
list函数很简单,把传入的每个参数插入到一个数组里,我们用bind()给list函数设置初始值,因为不用改变list中this的指向,所以直接传undefined,从第二个参数开始,就是要传入list函数的值,list2和list3的返回值很好的说明了一切。
我自己一般使用的bind()的场景是配合settimeout函数,因为在执行settimeout时,this会默认指向window对象,在使用bind()之前,我是这么做的:
function coder(name) {
var that = this;
that.name = name;
that.getname = function() {
console.log(that.name)
};
that.delaygetname = function() {
settimeout(that.getname,1000)
};
}
var me = new coder('jins')
me.delaygetname()//延迟一秒输出jins
在函数内顶层定义一个that缓存this的指针,这样不论怎么调用,that都是指向 coder的实例,但是多定义一个变量总是让人不太舒服。
使用bind()就简单多了:
function coder(name) {
this.name = name;
this.getname = function() {
console.log(this.name)
};
this.delaygetname = function() {
settimeout(this.getname.bind(this),1000)
};
}
var me = new coder('jins')
me.delaygetname()//延迟一秒输出jins
这样就ok了,直接把settimeout的this绑定到外层的this,这肯定是我们想要的!
更多关于function中的bind()示例详解。