美文网首页
响应事件处理

响应事件处理

作者: 打代码的小鱼姑娘 | 来源:发表于2018-07-02 16:32 被阅读0次

在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

        相关文章

        • iOS 响应链

          iOS开发 - 事件传递响应链iOS 响应者链,事件的传递事件传递之响应链Cocoa Touch事件处理流程--响...

        • 响应事件处理

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

        • iOS基础篇-事件处理

          1、首先需要理解iOS事件处理机制 理解事件处理、响应者、响应者链概念https://developer.appl...

        • 响应者链浅谈

          响应者对象 响应者对象(Response object) 响应者对象就是可以响应事件并对事件作出处理。iOS中UI...

        •  iOS触摸事件及响应者链条

          事件及响应者 响应者: 响应者对象是指能够处理事件的对象,即继承UIResponder的对象响应者链条:由很多响应...

        • vue.js 中的事件修饰符

          stop阻止事件传递(事件冒泡),即:事件处理只在当前元素的响应方法中响应,响应完毕后事件不继续传播给父元素; ...

        • iOS知识收集

          1. 响应者链 1.1 Cocoa Touch事件处理流程--响应者链 1.2 事件传递之响应链 多线程 http...

        • React 事件总线EventBus,实现全局事件响应

          如何将事件进行全局响应? 通过引入events 包进行全局事件响应 全局事件响应的好处 可处理深层次组件传值,以及...

        • 【OC梳理】事件传递响应链-实战篇(伪)

          接上篇,【OC梳理】事件传递响应链-原理篇对于事件传递响应链,最常见的处理就是处理不规则区域的点击事件(例如以前常...

        • 关于JavaScript跨浏览器事件的处理

          一、4种事件处理程序 事件就是用户或浏览器自身执行的某种操作,而事件处理程序即为响应某个事件的函数。事件处理程序大...

        网友评论

            本文标题:响应事件处理

            本文链接:https://www.haomeiwen.com/subject/diwkuftx.html