欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Javascript Function.prototype.bind详细分析

程序员文章站 2023-01-31 10:33:56
  function.prototype.bind分析 bind()方法会创建一个新的函数,成为绑定函数。当调用这个绑定函数时,绑定函数会以创建它时传入的第一个...

  function.prototype.bind分析

bind()方法会创建一个新的函数,成为绑定函数。当调用这个绑定函数时,绑定函数会以创建它时传入的第一个参数作为this,传入bind()方法的第二个以及以后的参数加上绑定函数运行时本身的参数按照顺序作为原函数的参数来调取原函数。

       实际使用中我们经常会碰到这样的问题:

var name = "pig";
function person(name){
  this.name = name;
  this.getname = function(){
    settimeout(function(){
      console.log("hello,my name is "+this.name);
    },100);
  }
}
var weiqi = new person("卫旗");
weiqi.getname();  
//hello,my name is pig

       这个时候输出this.name是pig,原因是this的指向是在运行函数时确定的,而不是在定义函数时确定的,再因为settimeout是在全局环境下只想,所以this就指向了window。

       以前解决这个问题的办法通常是缓存this,例如:

var name = "pig";
function person(name){
  this.name = name;
  this.getname = function(){
    //在这里缓存一个this
    var self = this;
    settimeout(function(){
      //在这里是有缓存this的self
      console.log("hello,my name is "+self.name);
    },100);
  }
}
var weiqi = new person("卫旗");
weiqi.getname();
//hello,my name is 卫旗

       这样就解决了这个问题,非常方便,因为它使得settimeout函数中可以访问person的上下文。

       现在有一个更好的解决办法,可以使用bind()函数,上面的例子可以被更新为:

var name = "pig";
function person(name){
  this.name = name;
  this.getname = function(){
    settimeout(function(){
      console.log("hello,my name is "+this.name);
    }.bind(this),100);
    //注意上面这一行,添加了bind(this)
  }
}
var weiqi = new person("卫旗");
weiqi.getname();
//hello,my name is 卫旗

       bind()最简单的用法是创建一个函数,使得这个函数无论怎么样调用都拥有同样的this值。javascript新手经常犯的一个错误就是将一个方法从一个对象中拿出来,然后再调用,希望方法中的this是原来的对象(比如在回调函数中传入这个方法)。如果不做特殊处理的话,一般会丢失原来的对象。从原来的函数和原来的对象创建一个绑定函数,则可以很漂亮的解决这个问题:

//定义全局变量x
var x = "window";
//在module内部定义x
var module = {
  x:"module",
  getx:function(){
    console.log(this.x);
  }
}
module.getx(); 
//返回module,因为在module内部调用getx()

var getx = module.getx;
getx();
//返回window,因为这个getx()是在全局作用域中调用的

//绑定getx()并将this值设为module
var boundgetx = getx.bind(module);
boundgetx();
//返回module,绑定以后this值始终为module

浏览器支持情况:

browser version support
chrome 7
firefox(gecko) 4.0(2)
internet explorer 9
opera 11.60
safari 5.14

       很不幸,function.prototype.bind在ie8及以下版本中不被支持,所以如果没有一个备选方案的话,可能会在运行时出现问题。bind函数在ecma-262第五版才被加入。它可能不无法在所有浏览器上运行。你可以在脚本部分加入如下代码,让不支持的浏览器也能使用bind()功能。

if (!function.prototype.bind) {
 function.prototype.bind = function (othis) {
  if (typeof this !== "function") {
   // closest thing possible to the ecmascript 5 internal iscallable 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;
 };
}

语法

    fun.bind(thisarg[, arg1[, arg2[, …]]])

参数

       thisarg,当绑定函数被调用时,该参数会作为原函数运行时的this指向,当使用new操作符调用绑定函数时,该参数无效。

       arg1, arg2, …,当绑定函数被调用时,这些参数加上绑定函数本身的参数会按照顺序作为原函数运行时的参数。

描述

       bind()函数会创建一个新的函数(一个绑定的函数)有同样的函数体(在ecmascript 5 规范内置call属性),当该函数(绑定函数的原函数)被调用时this值绑定到bind()的第一个参数,该参数不能被重写。绑定函数被调用时,bind()也接受预设的参数提供给原函数。一个绑定函数也能使用new操作符创建对象:这种行为就像把原函数当成构造器。提供的this值被忽略,同事调用的参数被提供给模拟函数。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!