React中ES5与ES6写法的区别总结

前言

相信很多React的初学者都被ES6的问题迷惑:各路大神都建议我们直接学习ES6的语法(class Foo extends React.Component),然而网上搜到的很多教程和例子都是ES5版本的,所以很多人在学习的时候连照猫画虎都不知道怎么做。所以这篇文章就整理了一些ES5和ES6的写法对照表,希望大家以后读到ES5的代码,也能通过对照,在ES6下实现相同的功能。下面话不多说了,来看看详细的介绍吧。

模块

引用

在ES5里,如果使用CommonJS标准,引入React包基本通过require进行,代码类似这样:

//ES5
var React = require("react");
var {
 Component,
 PropTypes
} = React; //引用React抽象组件

var ReactNative = require("react-native");
var {
 Image,
 Text,
} = ReactNative; //引用具体的React Native组件

在ES6里,import写法更为标准

//ES6
import React, {
 Component,
 PropTypes,
} from 'react';
import {
 Image,
 Text
} from 'react-native'

导出单个类

在ES5里,要导出一个类给别的模块用,一般通过module.exports来导出

//ES5
var MyComponent = React.createClass({
 ...
});
module.exports = MyComponent;

在ES6里,通常用export default来实现相同的功能:

//ES6
export default class MyComponent extends Component{
 ...
}

引用的时候也类似:

//ES5
var MyComponent = require('./MyComponent');

//ES6
import MyComponent from './MyComponent';

注意:导入和导出的写法必须配套,不能混用!

定义组件

在ES5里,通常通过React.createClass来定义一个组件类,像这样:

//ES5
var Photo = React.createClass({
 render: function() {
  return (
   <Image source={this.props.source} />
  );
 },
});

在ES6里,我们通过定义一个继承自React.Component的class来定义一个组件类,像这样:

//ES6
class Photo extends React.Component {
 render() {
  return (
   <Image source={this.props.source} />
  );
 }
}

给组件定义方法

从上面的例子里可以看到,给组件定义方法不再用 名字: function()的写法,而是直接用名字(),在方法的最后也不能有逗号了。

//ES5
var Photo = React.createClass({
 componentWillMount: function(){

 },
 render: function() {
  return (
   <Image source={this.props.source} />
  );
 },
});
//ES6
class Photo extends React.Component {
 componentWillMount() {

 }
 render() {
  return (
   <Image source={this.props.source} />
  );
 }
}

定义组件的属性类型和默认属性

在ES5里,属性类型和默认属性分别通过propTypes成员和getDefaultProps方法来实现

//ES5
var Video = React.createClass({
 getDefaultProps: function() {
  return {
   autoPlay: false,
   maxLoops: 10,
  };
 },
 propTypes: {
  autoPlay: React.PropTypes.bool.isRequired,
  maxLoops: React.PropTypes.number.isRequired,
  posterFrameSrc: React.PropTypes.string.isRequired,
  videoSrc: React.PropTypes.string.isRequired,
 },
 render: function() {
  return (
   <View />
  );
 },
});

在ES6里,可以统一使用static成员来实现

//ES6
class Video extends React.Component {
 static defaultProps = {
  autoPlay: false,
  maxLoops: 10,
 }; // 注意这里有分号
 static propTypes = {
  autoPlay: React.PropTypes.bool.isRequired,
  maxLoops: React.PropTypes.number.isRequired,
  posterFrameSrc: React.PropTypes.string.isRequired,
  videoSrc: React.PropTypes.string.isRequired,
 }; // 注意这里有分号
 render() {
  return (
   <View />
  );
 } // 注意这里既没有分号也没有逗号
}

也有人这么写,虽然不推荐,但读到代码的时候你应当能明白它的意思:

//ES6
class Video extends React.Component {
 render() {
  return (
   <View />
  );
 }
}
Video.defaultProps = {
 autoPlay: false,
 maxLoops: 10,
};
Video.propTypes = {
 autoPlay: React.PropTypes.bool.isRequired,
 maxLoops: React.PropTypes.number.isRequired,
 posterFrameSrc: React.PropTypes.string.isRequired,
 videoSrc: React.PropTypes.string.isRequired,
};

注意: 对React开发者而言,static成员在IE10及之前版本不能被继承,而在IE11和其它浏览器上可以,这有时候会带来一些问题。React Native开发者可以不用担心这个问题。

初始化STATE

ES5下情况类似,

//ES5
var Video = React.createClass({
 getInitialState: function() {
  return {
   loopsRemaining: this.props.maxLoops,
  };
 },
})

ES6下,有两种写法:

//ES6
class Video extends React.Component {
 state = {
  loopsRemaining: this.props.maxLoops,
 }
}

不过我们推荐更易理解的在构造函数中初始化(这样你还可以根据需要做一些计算):

//ES6
class Video extends React.Component {
 constructor(props){
  super(props);
  this.state = {
   loopsRemaining: this.props.maxLoops,
  };
 }
}

把方法作为回调提供

很多习惯于ES6的用户反而不理解在ES5下可以这么做:

//ES5
var PostInfo = React.createClass({
 handleOptionsButtonClick: function(e) {
  // Here, 'this' refers to the component instance.
  this.setState({showOptionsModal: true});
 },
 render: function(){
  return (
   <TouchableHighlight onPress={this.handleOptionsButtonClick}>
    <Text>{this.props.label}</Text>
   </TouchableHighlight>
  )
 },
});

在ES5下,React.createClass会把所有的方法都bind一遍,这样可以提交到任意的地方作为回调函数,而this不会变化。但官方现在逐步认为这反而是不标准、不易理解的。

在ES6下,你需要通过bind来绑定this引用,或者使用箭头函数(它会绑定当前scope的this引用)来调用

//ES6
class PostInfo extends React.Component
{
 handleOptionsButtonClick(e){
  this.setState({showOptionsModal: true});
 }
 render(){
  return (
   <TouchableHighlight
    onPress={this.handleOptionsButtonClick.bind(this)}
    onPress={e=>this.handleOptionsButtonClick(e)}
    >
    <Text>{this.props.label}</Text>
   </TouchableHighlight>
  )
 },
}

箭头函数实际上是在这里定义了一个临时的函数,箭头函数的箭头=>之前是一个空括号、单个的参数名、或用括号括起的多个参数名,而箭头之后可以是一个表达式(作为函数的返回值),或者是用花括号括起的函数体(需要自行通过return来返回值,否则返回的是undefined)。

// 箭头函数的例子
()=>1
v=>v+1
(a,b)=>a+b
()=>{
 alert("foo");
}
e=>{
 if (e == 0){
  return 0;
 }
 return 1000/e;
}

需要注意的是,不论是bind还是箭头函数,每次被执行都返回的是一个新的函数引用,因此如果你还需要函数的引用去做一些别的事情(譬如卸载监听器),那么你必须自己保存这个引用

// 错误的做法
class PauseMenu extends React.Component{
 componentWillMount(){
  AppStateIOS.addEventListener('change', this.onAppPaused.bind(this));
 }
 componentDidUnmount(){
  AppStateIOS.removeEventListener('change', this.onAppPaused.bind(this));
 }
 onAppPaused(event){
 }
}
// 正确的做法
class PauseMenu extends React.Component{
 constructor(props){
  super(props);
  this._onAppPaused = this.onAppPaused.bind(this);
 }
 componentWillMount(){
  AppStateIOS.addEventListener('change', this._onAppPaused);
 }
 componentDidUnmount(){
  AppStateIOS.removeEventListener('change', this._onAppPaused);
 }
 onAppPaused(event){
 }
}

从网上我们还学习到一种新的做法:

// 正确的做法
class PauseMenu extends React.Component{
 componentWillMount(){
  AppStateIOS.addEventListener('change', this.onAppPaused);
 }
 componentDidUnmount(){
  AppStateIOS.removeEventListener('change', this.onAppPaused);
 }
 onAppPaused = (event) => {
  //把方法直接作为一个arrow function的属性来定义,初始化的时候就绑定好了this指针
 }
}

Mixins

在ES5下,我们经常使用mixin来为我们的类添加一些新的方法,譬如PureRenderMixin

var PureRenderMixin = require('react-addons-pure-render-mixin');
React.createClass({
 mixins: [PureRenderMixin],

 render: function() {
  return <div className={this.props.className}>foo</div>;
 }
});

然而现在官方已经不再打算在ES6里继续推行Mixin,他们说:Mixins Are Dead. Long Live Composition

尽管如果要继续使用mixin,还是有一些第三方的方案可以用,譬如这个方案

不过官方推荐,对于库编写者而言,应当尽快放弃Mixin的编写方式,上文中提到Sebastian Markbåge的一段代码推荐了一种新的编码方式:

//Enhance.js
import { Component } from "React";

export var Enhance = ComposedComponent => class extends Component {
 constructor() {
  this.state = { data: null };
 }
 componentDidMount() {
  this.setState({ data: 'Hello' });
 }
 render() {
  return <ComposedComponent {...this.props} data={this.state.data} />;
 }
};
//HigherOrderComponent.js
import { Enhance } from "./Enhance";

class MyComponent {
 render() {
  if (!this.data) return <div>Waiting...</div>;
  return <div>{this.data}</div>;
 }
}

export default Enhance(MyComponent); // Enhanced component

用一个“增强函数”,来某个类增加一些方法,并且返回一个新类,这无疑能实现mixin所实现的大部分需求。

ES6+带来的其它好处

解构&属性延展

结合使用ES6+的解构和属性延展,我们给孩子传递一批属性更为方便了。这个例子把className以外的所有属性传递给div标签:

class AutoloadingPostsGrid extends React.Component {
  render() {
    var {
      className,
      ...others, // contains all properties of this.props except for className
    } = this.props;
    return (
      <div className={className}>
        <PostsGrid {...others} />
        <button onClick={this.handleLoadMoreClick}>Load more</button>
      </div>
    );
  }
}

下面这种写法,则是传递所有属性的同时,用覆盖新的className值:

<div {...this.props} className="override">
  …
</div>

这个例子则相反,如果属性中没有包含className,则提供默认的值,而如果属性中已经包含了,则使用属性中的值

<div className="base" {...this.props}>
  …
</div>

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • 深入浅析react native es6语法

    react native是直接使用es6来编写代码,许多新语法能提高我们的工作效率 解构赋值 var { StyleSheet,Text,View } = React; 这句代码是ES6 中新增的解构(Destructuring)赋值语句.准许你获取对象的多个属性并且使用一条语句将它们赋给多个变量. 上面的代码等价于: var StyleSheet = React.StyleSheet; var Text = React.Text; var View = React.View 再看几个例子,以前

  • react开发中如何使用require.ensure加载es6风格的组件

    其实用的babel,在浏览器端就应该可以加载,之前少了个default: require.ensure([],(require) => { let A = require('./a.js').default; }) 以下方式也可以,但是比较low,可以作废了: 1.问题提出:想通过require.ensure加载es6风格的模块? 2.出现问题:import方式本身就是静态设计方式.如果require进来的是commonjs模块或者amd则没问题,但项目只想es6一个书写风格,行吗? 遗憾的是:

  • 利用ES6语法重构React组件详解

    一.创建组件 ES6 class创建的组件语法更加简明,也更符合javascript.内部的方法不需要使用function关键字. React.createClass import React from 'react'; const MyComponent = React.createClass({ render: function() { return ( <div>以前的方式创建的组件</div> ); } }); export default MyComponent; Rea

  • ES6下React组件的写法示例代码

    本文主要跟大家分享了ES6下React组件的写法示例,下面来一起看看详细的介绍: 一:定义React组件 class Hello extends React.Component { render() { return <h1>Hello, {this.props.value}</h1>; } } 二:声明prop类型与默认prop class Hello extends React.Component { // ... } Hello.propTypes = { value: Re

  • React.js中常用的ES6写法总结(推荐)

    一 模块 1 引入模块以便使用 用import实现: import '模块文件地址' import 组件 from '模块文件地址' 2 导出模块 用export default实现: export default class MyComponent extends Component{ ... } 引用: import MyComponent from './MyComponent'; 二 组件 1 定义组件 通过定义一个继承自React.Component的class来定义一个组件类: cl

  • 深入理解React中es6创建组件this的方法

    首发于:https://mingjiezhang.github.io/. 在JavaScript中,this对象是运行时基于函数的执行环境(也就是上下文)绑定的. 从react中的demo说起 Facebook最近一次更新react时,将es6中的class加入了组件的创建方式当中.Facebook也推荐组件创建使用通过定义一个继承自 React.Component 的class来定义一个组件类.官方的demo: class LikeButton extends React.Component

  • React中ES5与ES6写法的区别总结

    前言 相信很多React的初学者都被ES6的问题迷惑:各路大神都建议我们直接学习ES6的语法(class Foo extends React.Component),然而网上搜到的很多教程和例子都是ES5版本的,所以很多人在学习的时候连照猫画虎都不知道怎么做.所以这篇文章就整理了一些ES5和ES6的写法对照表,希望大家以后读到ES5的代码,也能通过对照,在ES6下实现相同的功能.下面话不多说了,来看看详细的介绍吧. 模块 引用 在ES5里,如果使用CommonJS标准,引入React包基本通过re

  • ES5和ES6中类的区别总结

    类定义与调用的区别 在 ES5 中主要是通过构造函数方式和原型方式来定义一个类,但是在 ES6 新引入了 class 关键字,使之具有了正式类的能力,类(class)是ECMAScript 中新的基础性语法糖结构.虽然 ES6 类表面上看起来可以支持正式的面向对象编程,但实际上它背后使用的仍然是原型和构造函数的概念. 使用 ES5 定义一个类并调用 function Person(name, age, job) { this.name = "Totora"; this.age = 19

  • 分析ES5和ES6的apply区别

    目录 概述 函数签名 可选参数 非严格模式 异常处理 实际使用 总结 概述 众所周知, ES6 新增了一个全局.内建.不可构造的Reflect对象,并提供了其下一系列可被拦截的操作方法.其中一个便是Reflect.apply()了.下面探究下它与传统 ES5 的Function.prototype.apply()之间有什么异同. 函数签名 MDN 上两者的函数签名分别如下: Reflect.apply(target, thisArgument, argumentsList) function.a

  • JS实现集合的交集、补集、差集、去重运算示例【ES5与ES6写法】

    本文实例讲述了JS实现集合的交集.补集.差集.去重运算.分享给大家供大家参考,具体如下: ES5写法: ///集合取交集 Array.intersect = function () { var result = new Array(); var obj = {}; for (var i = 0; i < arguments.length; i++) { for (var j = 0; j < arguments[i].length; j++) { var str = arguments[i][

  • React组件的三种写法总结

    React 专注于 view 层,组件化则是 React 的基础,也是其核心理念之一,一个完整的应用将由一个个独立的组件拼装而成. 截至目前 React 已经更新到 v15.4.2,由于 ES6 的普及和不同业务场景的影响,我们会发现目前主要有三种创建 React 组件的写法:1. ES5写法React.createClass,2. ES6写法React.Component,3. 无状态的函数式写法(纯组件-SFC). 你们最钟爱哪种写法呢?萝卜青菜各有所爱~ 每个团队都有自己的代码规范和开发模

  • es6在react中的应用代码解析

    不论是React还是React-native,facebook官方都推荐使用ES6的语法,没在项目中使用过的话,突然转换过来会遇到一些问题,如果还没有时间系统的学习下ES6那么注意一些常见的写法暂时也就够用的,这会给我们的开发带来很大的便捷,你会体验到ES6语法的无比简洁.下面给大家介绍es6在react中的应用,具体内容如下所示: import React,{Component} from 'react'; class RepeatArrayextends Component{ constru

  • 详解javascript中var与ES6规范中let、const区别与用法

    随着ES6规范的到来,Js中定义变量的方法已经由单一的 var 方式发展到了 var.let.const 三种之多.var 众所周知,可那俩新来的货到底有啥新特性呢?到底该啥时候用呢?下面就是小编总结出的关于javascript中var与ES6规范中let.const区别详解 我们先来絮叨絮叨 var 方式定义变量有啥 bug ? Js没有块级作用域 请看这样一条规则:在JS函数中的var声明,其作用域是函数体的全部. for(var i=0;i<10;i++){ var a = 'a'; }

  • 说说react中引入css的方式有哪些并区别在哪

    目录 前言 方式 在组件内 组件中引入css文件 组件中引入 .module.css 文件 CSS in JS 区别 前言 组件式开发选择合适的css解决方案尤为重要 通常会遵循以下规则: 可以编写局部css,不会随意污染其他组件内的原生: 可以编写动态的css,可以获取当前组件的一些状态,根据状态的变化生成不同的css样式: 支持所有的css特性:伪类.动画.媒体查询等: 编写起来简洁方便.最好符合一贯的css风格特点 在这一方面,vue使用css起来更为简洁: 通过 style 标签编写样式

随机推荐