关于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高级程序设计》。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,519评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,842评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,544评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,742评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,646评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,027评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,513评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,169评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,324评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,268评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,299评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,996评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,591评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,667评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,911评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,288评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,871评论 2 341

推荐阅读更多精彩内容