响应事件处理

在react中,处理事件响应的方式主要有: (1). 使用匿名函数; (2).使用组件的方法; (3). 使用属性初始化的语法; 

    (1).使用匿名函数,

        这是最简单的一种方式,哪里需要响应事件就在哪里创建一个匿名函数,

先看代码: 

  handleChange(e){

        console.log("e",e);

  }

render(){

  return   <div>

                <Button onClick = {(e) => handleChange(e)}></Button>

        </div>

}

        点击Button的事件响应函数是一个匿名函数,这应该是最常见的处理事件响应的方式了。这种方式的好处是,简单直接。哪里需要处理事件响应,就在哪里定义一个匿名函数处理。

        在上面的代码中,也可以不使用箭头函数,直接使用onClick = {function(){}}的形式, 但实际开发中不会用到这种方式,因为箭头函数解决了this绑定的问题,可以将函数体内的this绑定到当前对象,而不是运行时调用函数的对象。如果响应函数中需要使用this.state,那么这种方式就无法正常运行了。所以项目中一般直接使用箭头函数定义的匿名函数作为事件响应。

        使用匿名函数的缺点是:当事件响应逻辑比较复杂时,匿名函数的代码量会很大,会导致render函数变得臃肿,不容易直观地看出组件最终渲染出的元素结构。另外,每次render方法调用时,都会重新创建一个匿名函数对象,带来额外的性能开销,当组件的层级越低时,这种开销就越大,因为任何一个上层组件的变化都可能会触发这个组件的render方法。当然,在大多数情况下,这点性能损失是可以不必在意的。

(2).使用组件方法:

    毫无疑问,按钮的响应事件是直接调用组件的方法,这种方式需要通过在 constructor中手动绑定this;

    看代码:

    constructor(props, context) {

  super(props, context);

  this.state = {

    options: {},

  }

  this.handleChange = this.handleChange.bind(this);

}

handleChange (e){

  console.log("e",e);

}

render(){

return <div>

        <button onClick={this.handleChange }>功能介绍听您的</button>

</div>

}

        点击Button的事件响应函数是组件的方法:handleClick。这种方式的好处是:每次render方法的调用,不会重新创建一个新的事件响应函数,没有额外的性能损失。但是,使用这种方式要在构造函数中为作为事件响应的方法(handleClick),手动绑定this: this.handleClick = this.handleClick.bind(this),这是因为ES6 语法的缘故,ES6 Class 的方法默认不会把this绑定到当前的实例对象上,需要我们手动绑定。每次都手动绑定this是不是有点繁琐?好吧,让我们来看下一种方式。

(3). 使用属性初始化语法

    //代码4

    class MyComponent extends React.Component{

        state = {number: 0};

  handleClick = () => {

    this.setState({

      number: ++this.state.number

    });

  }

  render() {

    return (

<button onClick={this.handleClick}>功能介绍听您的</button>

    );

  }

}

说明: 这里是利用的es7的属性初始化语法, 从constructor里拿出来放到property initializer(属性初始化器)里。我们可以看到这里没有使用constructor函数, 直接将state写在了外面. constructor是ES6中类的构造函数。整个构造函数需要一个参数props,其全部的值都在上文的代码中给出:title、image等。state = ...一句中,使用props初始化了整个组件的state初值。


扩展:事件响应函数的传参问题

事件响应函数默认是会被传入一个事件对象Event作为参数的。如果想传入其他参数给响应函数应该怎么办呢?使用第一种方式的话很简单,直接使用新参数:

//代码5

class MyComponent extends React.Component{

  constructor(props) {

    super(props);

    this.state = {

      list: [1,2,3,4],

      current: 1    };

  }

  handleClick(item,event) {

    this.setState({

      current: item

    });

  }

  render() {

    return (

            {this.state.list.map(

              (item)=>(

                onClick={(event) => this.handleClick(item, event)}>{item}

              )

            )}

        );

      }

    }

    onClick的响应函数中,方法体内可以直接使用新的参数item。

    使用第二种方式的话,可以把绑定this的操作延迟到render中,在绑定this的同时,绑定额外的参数:

    //代码6

    class MyComponent extends React.Component{

      constructor(props) {

        super(props);

        this.state = {

          list: [1,2,3,4],

          current: 1    };

      }

      handleClick(item) {

        this.setState({

          current: item

        });

      }

      render() {

        return (

              {this.state.list.map(

                (item)=>(

                  onClick={this.handleClick.bind(this, item)}>{item}

                )

              )}

          );

        }

      }

      使用第三种方式,解决方案和第二种基本一致:

      //代码7

      class MyComponent extends React.Component{

        constructor(props) {

          super(props);

          this.state = {

            list: [1,2,3,4],

            current: 1    };

        }

        handleClick = (item) =>  {

          this.setState({

            current: item

          });

        }

        render() {

          return (

                {this.state.list.map(

                  (item)=>(

                    onClick={this.handleClick.bind(undefined, item)}>{item}

                  )

                )}

            );

          }

        }

                不过这种方式就有点鸡肋了,因为虽然你不需要通过bind函数绑定this,但仍然要使用bind函数来绑定其他参数。

                关于事件响应函数,还有一个地方需要注意。不管你在响应函数中有没有显式的声明事件参数Event,React都会把事件Event作为参数传递给响应函数,且参数Event的位置总是在其他自定义参数的后面。例如,在代码6和代码7中,handleClick的参数中虽然没有声明Event参数,但你依然可以通过arguments[1]获取到事件Event对象。

                总结一下,三种事件处理的方式,第一种有额外的性能损失;第二种需要手动绑定this,代码量增多;第三种用到了ES7的特性,目前并非默认支持,需要Babel插件的支持,但是写法最为简洁,也不需要手动绑定this。一般推荐使用第二种和第三种方式。

        参考文档:  1. https://www.jianshu.com/p/429b2326bf9b

        2.https://www.cnblogs.com/ikcamp/p/8989492.html

        3. 官方文档:http://www.css88.com/react/docs/handling-events.html

        4. https://segmentfault.com/q/1010000007247736/a-1020000007247989

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