关于js函数的this和bind(),apply(),call()方法

作为一名差生,我一直没有系统的了解js的apply(),call(),bind()方法。之前也找到一些网文介绍这几个的,但是说真的,以我差生的水平看求不懂。

直到昨天我自己写代码时遇到和标题有关的问题,正巧前段时间失业在家系统看了下《JavaScript高级程序设计(第3版)高清完整PDF中文》,我觉得有必要把自己困惑的发生和解决写下来,来帮助需要帮助的后来者,而不是让后来者《JavaScript 从开始到放弃》。当然了,这是个玩笑。

按自己的理解写的,也许会有错误,欢迎指正。

在下面的例子里,遇到了和bind this有关的问题(以下为正确代码):

import React, { Component } from 'react';
...//省略代码
import NewsList from './components/news-list/'
import InfiniteScroll from 'react-infinite-scroller';

class App extends Component {
  constructor(props) {
      super(props);
      this.state = {
        ....//省略代码
        offset:0,
        limit:10,
        ....//省略代码
      };
    }
  componentDidMount(){
    this.loadMore()
  }
  async loadMore() {
    let data = await request(this.state.offset, this.state.limit)
     ...//省略代码
  }
  render() {
    return (
      <div className="App">
        <InfiniteScroll
//8行      pageStart={0}
//9行      loadMore={this.loadMore.bind(this)} //注意这里的bind(this)
//10行     hasMore={this.state.more}>
            <NewsList data={this.state.newsList}/>
        </InfiniteScroll>
      </div>
    );
  }
}

export default App;

倒数第9行里,组件InfiniteScroll的loadMore属性需要一个函数,最开始我是这样写的:

...
loadMore={this.loadMore}
...

但是在跑例子的时候,系统报错找不到offset:

App.js:24 Uncaught (in promise) TypeError: Cannot read property 'offset' of undefined

奇了怪,我的loadMore函数里明明定义了offset。

为了解决上述问题,有幸读到Javascript高级程序设计(第3版),里面写的非常详细:

this

《javascript高级程序设计》这本书里的[匿名函数]这一章也有讲到,摘抄如下:this对象是在运行时基于函数的执行环境绑定的:在全局函数中,this等于window,而当函数被作为某个对象的方法调用时,this等于那个对象。不过,匿名函数的执行环境具有全局性,因为其this对象通常指向window。

也就是说,函数执行时,this 总是指向调用该函数的对象,规则可以简单概括如下:
有对象就指向调用对象
没调用对象就指向全局对象
用new构造就指向新对象
通过 apply 或 call 或 bind 可以改变this的指向

来看下面的例子。

window.color = "red"
var o = { color: "blue"};

function sayColor(){
    alert(this.color); 
}

sayColor();        //"red"

o.sayColor = sayColor;
o.sayColor();      //"blue"

上面的函数sayColor()是在全局作用域中定义的,它引用了this对象。由于在调用函数之前,this的值并不确定,因此this可能会在代码执行过程中引用不同的对象。当在全局作用域中调用sayColor()时,this引用的是全局对象 window;换句话说,对 this.color 求值会转换成对window.color 求值,于是结果就返回了"red"。而当把这个函数赋给对象 o 并调用 o.sayColor() 时,this引用的是对象o,因此对 this.color 求值会转换成对 o.color 求值,结果就返回了"blue"。

回到之前的InfiniteScroll调用loadMore报错的问题,就明白了,this的值是在InfiniteScroll执行的时候才确定的,此时的this引用的是InfiniteScroll对象,而InfiniteScroll对象里可能并没有定义offset,所以报了错。

那这个时候如何解决this引用错误的问题呢?有请:

bind()

JavaScript高级程序设计里介绍:bind()方法会创建一个函数的实例,其 this 值会被绑定到传给bind()函数的值。例如:

window.color = "red";
var o = { color: "blue"};

function sayColor(){
      alert(this.color);
}
var objectSayColor = sayColor.bind(o);
objectSayColor();      //blue

在这里,sayColor()调用 bind() 并传入对象 o,创建了 objectSayColor()函数。objectSayColor()函数的 this 值等于o,因此即使是在全局作用域中调用这个函数,也会看到"blue"。

又回到之前的InfiniteScroll,我们在

...
//错误
loadMore={this.loadMore}
...
...
//正确
loadMore={this.loadMore.bind(this)}
...

this.loadMore函数在调用bind时,会先创建this.loadMore函数的一个实例,而此时的创建实例时的this是引用App的,所以之后this.loadMore不管怎么执行,它里面的this始终是指向App,而App里定义了offset,所以不报错,并让结果达到了期望。

Bind的其他用法,请见:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Function/bind

再来说说

apply()和call()

为什么一起说,因为这俩方法的作用相同,区别仅在于接受参数的方式不同。这两个方法的用途都是在特定的作用域中调用函数,实际上等于设置函数体内this对象的值。
apply()方法接收两个参数,一个是在其中运行函数的作用域,另一个是参数数组;
call()方法接收多参数,第一个是在其中运行函数的作用域,剩余参数都直接传递给函数。
看起来apply()、call()和bind()函数的目的都一样,最核心都是为了改变函数的this引用。只不过调用了apply()、call()后函数就直接执行结果了。

    // 函数的apply、call方法,影响函数执行的作用域
    window.color = "red"
    var o = {
        color:'blue'
    }
    function sayColor(){
        console.log(this.color)
    }
    sayColor()//red
    sayColor.apply(o)     //blue
    sayColor.call(o)     //blue
    // bind方法,创建一个函数的实例,这个实例的this值会被绑定到传给bind()函数的值
    var objecSaycolor = sayColor.bind(o)//bind后,新函数objecSaycolor的this值变成了o
    objecSaycolor()     //blue

关于bind()详见高程3 页码第118页(实际137页)。
关于apply()、call()详见高程3 ]第116页(实际135页)。

最后,墙裂建议大家去读一读《Javascript高级程序设计》。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容