React学习笔记—JSX

作者: 徐国军_plus | 来源:发表于2018-02-27 00:43 被阅读41次

    所谓JSX,是JavaScript的语法扩展(eXtension),让我们在JavaScript中可以编写像HTML一样的代码。

    JSX中的这几段代码看起来和HTML几乎一摸一样,都可以使用<div><button>之类的元素,所以只要熟悉HTML,学习JSX完全不成问题,但是,我们一定要明白两者的不同之处。

    首先,在JSX中使用的“元素”不局限于HTML中的元素,可以是任何一个React组件。例如:

    // Counter组件
    import React, { Component } from 'react';
    
    class Counter extends Component {
    
      constructor(props) {
        super(props);
        this.onClickIncrementButton = this.onClickIncrementButton.bind(this);
        this.onClickDecrementButton = this.onClickDecrementButton.bind(this);
        this.state = {
          count: props.initValue
        }
      }
    
      onClickIncrementButton() {
        this.setState({count: this.state.count + 1});
      }
    
      onClickDecrementButton() {
        this.setState({count: this.state.count - 1});
      }
    
      render() {
        const {caption} = this.props; 
        console.log(caption)
        return (
          <div>
            <button style={buttonStyle} onClick={this.onClickIncrementButton}>+</button>
            <button style={buttonStyle} onClick={this.onClickDecrementButton}>-</button>
            <span>{caption} count: {this.state.count}</span>
          </div>
        );
      }
    }
    
    Counter.defaultProps = {
      initValue: 0
    };
    
    export default Counter;
    
    // ControlPanel组件
    import React, { Component } from 'react';
    import Counter from './Counter.js';
    
    class ControlPanel extends Component {
      render() {
        return (
          <div style={style}>
            <Counter caption="First"/> 
            <Counter caption="Second" initValue={10} />
            <Counter caption="Third" initValue={20} />
          </div>
        );
      }
    }
    
    export default ControlPanel;
    

    在ControlPanel组件中可以看到,创建的Counter组件被直接应用在了JSX中,使用方法和其他元素一样,这一点是传统的HTML做不到的。

    React判断一个元素是HTML元素还是React组件的原则就是看第一个字母是否大写,如果在JSX中我们不使用Counter而是使用counter就得不到想要的结果。

    其次,在JSX中可以通过onClick这样的方式给一个元素添加一个事件处理函数,当然,在HTML中也可以用onclick(注意和onClick拼写有区别),但在HTML中直接书写onclick一直就是为人诟病的写法,网页应用开发界面一直倡导的是用jQuery的方法添加事件处理函数,直接写onclick会带来代码混乱的问题。

    这就带来一个问题,既然长期以来不倡导在HTML中使用onclick,为什么在React的JSx中我们却要使用onclick这样的方式来添加事件处理函数呢?

    在React出现之初,很多人对React这样的设计非常反感,因为React把类似HTML的标记语言和JavaScript混在一起了,但是,随着时间的推移,业界逐渐认可了这种方式,因为大家都发现,以前用HTML来代表内容,用CSS代表样式,用JavaScript来定义交互行为,这三种语言分在三种不同的文件里面,实际上是把不同技术分开管理了,而不是逻辑上的“分而治之”。

    根据做同一件事的代码应该有高耦合性的设计原则,既然我们要实现一个Counter,为什么不把实现这个功能的所有代码集中在一个文件里呢?

    那么,JSX中使用onClick添加事件处理函数,是否代表网页应用开发兜了一个大圈,最终回到了起点了呢?

    不是这样,在JSX中使用onClick添加事件处理方式和onclick有很大不同。
    即使现在,我们还是要说在HTML中直接使用onclick很不专业,原因如下:

    1. onclick添加的事件处理函数是在全局环境下执行的,这污染了全局环境,很容易产生意料不到的后果;

    2. 给很多DOM元素添加onclick事件,可能会影响网页的性能,毕竟,网页需要的事件处理函数越多,性能就越低。

    3. 对于onclick的DOM元素,如果要动态地从DOM树种删掉的话,需要把对应的事件处理函数注销,假如忘了注销,就可能造成内存泄漏,这样的bug很难被发现。

    上面说的这些问题,在JSX中都不存在。

    首先,onClick挂载的每个函数,都可以控制在组件范围内,不会污染全局空间。以上面的Counter组件为例:


    image.png

    我们在Counter的JSX中使用了onClick,但并没有产生直接使用的onclick(注意是onclick不是onClick)的HTML,而是使用了事件委托(event delegation)的方式处理点击事件,无论有多少个onClick出现,其实最后都只在DOM树上添加了一个事件处理函数,挂在最顶层的DOM节点上。所有的点击事件都被这个事件处理捕获,然后根据具体组件分配给特定函数,使用事件委托的性能当然要比每个onClick都挂载一个事件处理函数要高。

    因为React控制了组件的生命周期,在unmount的时候自然能够清除相关的所有事件处理函数,内存泄漏也不再是一个问题。

    除了在组件中定义交互行为,我们还可以在React组件中定义样式,我们可以修改Counter组件中的render函数,代码如下:

    import React, { Component} from 'react';
    
    const buttonStyle = {
      margin: '10px'
    };
    
    class Counter extends Component {
    
      constructor(props) {
        super(props);
        this.onClickIncrementButton = this.onClickIncrementButton.bind(this);
        this.onClickDecrementButton = this.onClickDecrementButton.bind(this);
    
        this.state = {
          count: props.initValue
        }
      }
    
      onClickIncrementButton() {
        this.setState({count: this.state.count + 1});
      }
    
      onClickDecrementButton() {
        this.setState({count: this.state.count - 1});
      }
    
      render() {
        const {caption} = this.props; 
        console.log(caption)
        return (
          <div>
            <button style={buttonStyle} onClick={this.onClickIncrementButton}>+</button>
            <button style={buttonStyle} onClick={this.onClickDecrementButton}>-</button>
            <span>{caption} count: {this.state.count}</span>
          </div>
        );
      }
    }
    
    Counter.defaultProps = {
      initValue: 0
    };
    
    export default Counter;
    

    可以看到React的组件可以把JavaScript、HTML和CSS的功能集中在一个文件中,实现真正的组件封装。

    相关文章

      网友评论

        本文标题:React学习笔记—JSX

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