call和apply的使用和区别不再做阐述,可以参考我的另一篇随笔《JavaScript中call和apply方法的使用》(https://www.cnblogs.com/lcr-smg/p/10067398.html),这里只是针对bind的用法及与call和apply两者的区别。
bind的用法
bind() 方法与 apply 和 call 很相似,也是可以改变函数体内 this 的指向。
MDN的解释是:bind()方法会创建一个新函数,称为绑定函数,当调用这个绑定函数时,绑定函数会以创建它时传入 bind()方法的第一个参数作为 this,传入 bind() 方法的第二个以及以后的参数加上绑定函数运行时本身的参数按照顺序作为原函数的参数来调用原函数。直接来看看具体如何使用,在常见的单体模式中,通常我们会使用 _this , that , self 等保存 this ,这样我们可以在改变了上下文之后继续引用到它。 像这样:
1 var foo = { 2 bar : 1, 3 eventBind: function(){ 4 var _this = this; 5 $('.someClass').on('click',function(event) { 6 /* Act on the event */ 7 console.log(_this.bar); //1 8 }); 9 } 10 }
由于 Javascript 特有的机制,上下文环境在 eventBind:function(){ } 过渡到 $('.someClass').on('click',function(event) { }) 发生了改变,上述使用变量保存 this 这些方式都是有用的,也没有什么问题。当然使用 bind() 可以更加优雅的解决这个问题:
1 var foo = { 2 bar : 1, 3 eventBind: function(){ 4 $('.someClass').on('click',function(event) { 5 /* Act on the event */ 6 console.log(this.bar); //1 7 }.bind(this)); 8 } 9 }
在上述代码里,bind() 创建了一个函数,当这个click事件绑定在被调用的时候,它的 this 关键词会被设置成被传入的值(这里指调用bind()时传入的参数)。因此,这里我们传入想要的上下文 this(其实就是 foo ),到 bind() 函数中。然后,当回调函数被执行的时候, this 便指向 foo 对象。再来一个简单的栗子:
1 var bar = function(){ 2 console.log(this.x); 3 } 4 var foo = { 5 x:3 6 } 7 bar(); // undefined 8 var func = bar.bind(foo); 9 func(); // 3
这里我们创建了一个新的函数 func,当使用 bind() 创建一个绑定函数之后,它被执行的时候,它的 this 会被设置成 foo , 而不是像我们调用 bar() 时的全局作用域。有个有趣的问题,如果连续 bind() 两次,亦或者是连续 bind() 三次那么输出的值是什么呢?像这样:
1 var bar = function(){ 2 console.log(this.x); 3 } 4 var foo = { 5 x:3 6 } 7 var sed = { 8 x:4 9 } 10 var func = bar.bind(foo).bind(sed); 11 func(); //? 12 13 var fiv = { 14 x:5 15 } 16 var func = bar.bind(foo).bind(sed).bind(fiv); 17 func(); //?
答案是,两次都仍将输出 3 ,而非期待中的 4 和 5 。原因是,在Javascript中,多次 bind() 是无效的。更深层次的原因, bind() 的实现,相当于使用函数在内部包了一个 call / apply ,第二次 bind() 相当于再包住第一次 bind() ,故第二次以后的 bind 是无法生效的。
apply、call、bind比较
那么 apply、call、bind 三者相比较,之间又有什么异同呢?何时使用 apply、call,何时使用 bind 呢。简单的一个栗子:
1 var obj = { 2 x: 81, 3 }; 4 5 var foo = { 6 getX: function() { 7 return this.x; 8 } 9 } 10 11 console.log(foo.getX.bind(obj)()); //81 12 console.log(foo.getX.call(obj)); //81 13 console.log(foo.getX.apply(obj)); //81
三个输出的都是81,但是注意看使用 bind() 方法的,他后面多了对括号。
也就是说,区别是,当你希望改变上下文环境之后并非立即执行,而是回调执行的时候,使用 bind() 方法。而 apply/call 则会立即执行函数。
总结:
- apply 、 call 、bind 三者都是用来改变函数的this对象的指向的;
- apply 、 call 、bind 三者第一个参数都是this要指向的对象,也就是想指定的上下文;
- apply 、 call 、bind 三者都可以利用后续参数传参;
- bind 是返回对应函数,便于稍后调用;apply 、call 则是立即调用 。