前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >react的事件处理为什么要bind this 改变this的指向?

react的事件处理为什么要bind this 改变this的指向?

作者头像
Tz一号
发布2020-09-10 00:32:42
1.3K0
发布2020-09-10 00:32:42
举报
文章被收录于专栏:Tz一号Tz一号

react的事件处理会丢失this,所以需要绑定,为什么会丢失this?

首先来看摘自官方的一句话:

代码语言:javascript
复制
You have to be careful about the meaning of this in JSX callbacks. In JavaScript, class methods are not bound by default.

这句话大概意思就是,你要小心jax回调函数里面的this,class方法默认是不会绑定它的

让我十分疑惑,在我的知识范围理解中,class是es6里面新增的方法,不就用来继承原有对象上的属性和方法创建新的对象吗?就是代替原来的构造函数的一种更清晰的方式,为什么就不会绑定this呢?

可是查阅了一些es6的文档,并不是这样的啊,和class方法没啥关系吧,为什么要它背锅呢?

代码语言:javascript
复制
class Toggle extends React.Component {
  constructor(props) {
    super(props);
    this.state = {isToggleOn: true};

    // This binding is necessary to make `this` work in the callback
    this.handleClick = this.handleClick.bind(this);
  }

  handleClick() {
    this.setState(prevState => ({
      isToggleOn: !prevState.isToggleOn
    }));
  }

  render() {
    return (
      <button onClick={this.handleClick}> //这里调用的this也能拿到啊??
        {this.state.isToggleOn ? 'ON' : 'OFF'} //这里的this为什么没问题?
      </button>
    );
  }
}

这是官网上的一段代码,如果是是因为class的关系,handleClick里面拿不到this,那为什么render里面能拿到this,所以和class根本没关系吧本来就能拿到,那问题出现在哪里,为什么拿不到?

先看看解决办法

第一种,在constructor里面用bind绑定this

代码语言:javascript
复制
constructor(props) {
    super(props);
    this.state = {isToggleOn: true};

    // This binding is necessary to make `this` work in the callback
    this.handleClick = this.handleClick.bind(this);
  }

第二种,声明方法的时候使用箭头函数

代码语言:javascript
复制
  handleClick = () => {
    this.setState(prevState => ({
      isToggleOn: !prevState.isToggleOn
    }));
  }

第三种,调用的时候使用箭头函数

代码语言:javascript
复制
render() {
    return (
      <button onClick={ () => { this.handleClick } }> 
        {this.state.isToggleOn ? 'ON' : 'OFF'} 
      </button>
    );
  }

  这个时候我想起了原生dom绑定click的方法

代码语言:javascript
复制
<button onclick ="handleClick()">点我</button>

  两者比较,我发现了个区别,原生的绑定方法事件名后面多了个() 于是我尝试着在react里面的事件加一个()

代码语言:javascript
复制
render() {
    return (
      <button onClick={ this.handleClick() }> 
        {this.state.isToggleOn ? 'ON' : 'OFF'} 
      </button>
    );
  }

就像上面这样,然后我发现,无论我怎么点,都不会触发这个方法了,再细心点,就发现,在渲染的时候,就调用了一次,而且仅此一次,再也不能调用了.

原因是jsx语法,渲染的时候会把{}里面包裹的代码先解析一遍,因为如果加了括号,直接就执行了里面的函数,就没有东西了,但是这个时候,this是可以拿到的

代码语言:javascript
复制
class App extends Component {
  handleClick(){
      console.log(this); //下面调用加了(),这个时候发现,this是可以拿到的
  }
  render() {
    return (
      <div className="App">
        <button onClick={this.handleClick()}>点我</button> //这里加了括号的
      </div>
    );
  }
}

  好像问题越来越明朗了,为啥会拿不到,和class没有关系,完全是因为react自己封装的东西,先会把{}里面的代码解析一遍,于是大概就是下面这种情况了

代码语言:javascript
复制
const obj = {
        num:1
    }
    obj.handleClick = function () {
        console.log(this);
    }
    console.log(eval(obj.handleClick ));  // f(){ console.log(this) } react对{}的解析
    (eval(obj.handleClick))() //onclick触发点击事件 这里输出this是window,所以就等于丢失了this指向

    console.log(eval(() => { obj.handleClick() }));  // () => { obj.handleClick() } react对{}的解析
    (eval(() => {obj.handleClick()}))() //onclick触发点击事件 这里输出this还是obj,所以this就保留了

  所以问题出在react对{}的解析会把this的指向解除了

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2019-07-24 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • react的事件处理会丢失this,所以需要绑定,为什么会丢失this?
    • 首先来看摘自官方的一句话:
    • 第一种,在constructor里面用bind绑定this
    • 第二种,声明方法的时候使用箭头函数
    • 第三种,调用的时候使用箭头函数
      •   所以问题出在react对{}的解析会把this的指向解除了
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档