理解javascript中的Function.prototype.bind的方法


Posted in Javascript onFebruary 03, 2017

在初学Javascript时,我们也许不需要担心函数绑定的问题,但是当我们需要在另一个函数中保持上下文对象this时,就会遇到相应的问题了,我见过很多人处理这种问题都是先将this赋值给一个变量(比如self、_this、that等),尤其是var that = this是我见的最多的,这样当你改变环境之后就可以使用它。这些都是可以的,但是还有一种更好的、更专有的方法,那就是使用Function.prototype.bind,下面进行详尽的讲解。

第一部分:需要解决的问题

首先看下面的代码

var myObj = {

  specialFunction: function () {

  },

  anotherSpecialFunction: function () {

  },

  getAsyncData: function (cb) {
    cb();
  },

  render: function () {
this.getAsyncData(function () {
      this.specialFunction();
      this.anotherSpecialFunction();
    });
  }
};

myObj.render();

这里我希望创建一个对象,包含了前面两个普通的方法;第三个方法可以传递一个函数,传入的这个函数立即执行;最后一个方法会调用myObj对象的getAsyncData方法,这里使用了this,然后在getAsyncData方法中传入了一个函数,这个函数继续调用这个对象的前两个方法,仍使用了this,这时很多人实际上就可以看出问题所在了,将上述代码输入控制台,得到下面的结果:

TypeError: this.specialFunction is not a function

第二部分:问题剖析

在对象中render方法中的this的确是指向myObj对象的,所以我们可以通过this.getAsyncData来调用这个对象中的函数,但是当我们给其传递函数作为参数时,这里的this就指向了全局环境window了,因为全局环境中没有对象中的前两个方法,所以才会报错。

第三部分:解决问题的几种方式

所以我们需要做的就是正确调用对象中的前两个方法 ,很多人使用的方法便是首先在对象的环境中获取this赋值给另一个变量,这时就可以在后面的环境中调用了,如下所示:

render: function () {
    var that = this;
    this.getAsyncData(function () {
      that.specialFunction();
      that.anotherSpecialFunction();
    });
  }

虽然这种方法是可行的,但是使用Function.prototype.bind()会使代码更清晰、易懂,如下所示:

render: function () {

  this.getAsyncData(function () {

    this.specialFunction();

    this.anotherSpecialFunction();

  }.bind(this));

}

这里我们就成功地把this绑定到了环境中。

下面是另外一个简单的例子:

var foo = {
  x: 3
}

var bar = function(){
  console.log(this.x);
}

bar(); // undefined

var boundFunc = bar.bind(foo);

boundFunc(); // 3

下面的例子也是常见的:

this.x = 9;  // this refers to global "window" object here in the browser
var module = {
 x: 81,
 getX: function() { return this.x; }
};

module.getX(); // 81

var retrieveX = module.getX;
retrieveX();  
// returns 9 - The function gets invoked at the global scope

// Create a new function with 'this' bound to module
// New programmers might confuse the
// global var x with module's property x
var boundGetX = retrieveX.bind(module);
boundGetX(); // 81

第四部分:浏览器支持

但是这个方法在IE8及以下是不被支持的,所以我们可以使用MDN提供的方法来使得IE低版本支持.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,
                aArgs.concat(Array.prototype.slice.call(arguments)));
    };

  fNOP.prototype = this.prototype;
  fBound.prototype = new fNOP();

  return fBound;
 };
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScript高级程序设计 错误处理与调试学习笔记
Sep 10 Javascript
JavaScript面向对象知识串结(读JavaScript高级程序设计(第三版))
Jul 17 Javascript
jquery方法+js一般方法+js面向对象方法实现拖拽效果
Aug 30 Javascript
firefox下jquery iframe刷新页面提示会导致重复之前动作
Dec 17 Javascript
jQuery 动态云标签插件
Nov 11 Javascript
JavaScript保存并运算页面中数字类型变量的写法
Jul 06 Javascript
BootStrap智能表单demo示例详解
Jun 13 Javascript
js仿QQ邮箱收件人选择与搜索功能
Feb 10 Javascript
Vue 进阶教程之v-model详解
May 06 Javascript
深入理解JS中Number(),parseInt(),parseFloat()三者比较
Aug 24 Javascript
浅谈一个webpack构建速度优化误区
Jun 24 Javascript
ES11屡试不爽的新特性,你用上了几个
Oct 21 Javascript
JavaScript数组复制详解
Feb 02 #Javascript
常用jQuery选择器汇总
Feb 02 #Javascript
JavaScript优化以及前段开发小技巧
Feb 02 #Javascript
JavaScript字符集编码与解码详谈
Feb 02 #Javascript
JS实现购物车特效
Feb 02 #Javascript
jQuery实现复选框的全选和反选
Feb 02 #Javascript
jQuery制作图片旋转效果
Feb 02 #Javascript
You might like
PHP4.04简明安装
2006/10/09 PHP
可以在线执行PHP代码包装修正版
2008/03/15 PHP
ThinkPHP3.1新特性之G方法的使用
2014/06/19 PHP
PHP速成大法
2015/01/30 PHP
PHP7使用ODBC连接SQL Server2008 R2数据库示例【基于thinkPHP5.1框架】
2019/05/06 PHP
Jquery创建层显示标题和内容且随鼠标移动而移动
2014/01/26 Javascript
Javascript模拟加速运动与减速运动代码分享
2014/12/11 Javascript
javascript编写实用的省市选择器
2015/02/12 Javascript
JQuery跳出each循环的方法
2015/04/16 Javascript
javascript从定义到执行 你不知道的那些事
2016/01/04 Javascript
深入理解bootstrap框架之入门准备
2016/10/09 Javascript
利用jQuery对无序列表排序的简单方法
2016/10/16 Javascript
vue路由跳转时判断用户是否登录功能的实现
2017/10/26 Javascript
Vue中this.$router.push参数获取方法
2018/02/27 Javascript
详解Webpack-dev-server的proxy用法
2018/09/08 Javascript
深入学习JavaScript中的bom
2019/05/27 Javascript
JS实现滚动条触底加载更多
2019/09/19 Javascript
vue 使用vant插件做tabs切换和无限加载功能的实现
2020/11/04 Javascript
python实现简单socket程序在两台电脑之间传输消息的方法
2015/03/13 Python
Python HTML解析模块HTMLParser用法分析【爬虫工具】
2019/04/05 Python
Python3如何对urllib和urllib2进行重构
2019/11/25 Python
六种酷炫Python运行进度条效果的实现代码
2020/07/17 Python
序列化Python对象的方法
2020/08/01 Python
python dict如何定义
2020/09/02 Python
浅谈HTML5 服务器推送事件(Server-sent Events)
2017/08/01 HTML / CSS
全球领先的全景影像品牌:Insta360
2019/08/21 全球购物
PHP如何设置和取得Cookie值
2015/06/30 面试题
受欢迎的大学生自我评价
2013/12/05 职场文书
后勤主管工作职责
2013/12/07 职场文书
简历中自我评价范文3则
2013/12/14 职场文书
新学期国旗下演讲稿
2014/05/08 职场文书
2014年教师节红领巾广播稿
2014/09/10 职场文书
技术员岗位职责
2015/02/04 职场文书
学子宴致辞大全
2015/07/27 职场文书
Python如何把不同类型数据的json序列化
2021/04/30 Python
Vue鼠标滚轮滚动切换路由效果的实现方法
2021/08/04 Vue.js