react组件设计原则_React组件设计技巧

React组件设计

组件分类

展示组件和容器组件

展示组件

容器组件

关注事物的展示

关注事物如何工作

可能包含展示和容器组件,并且一般会有DOM标签和css样式

可能包含展示和容器组件,并且不会有DOM标签和css样式

常常允许通过this.props.children传递

提供数据和行为给容器组件或者展示组件

对第三方没有任何依赖,比如store 或者 flux action

调用flux action 并且提供他们的回调给展示组件

不要指定数据如何加载和变化

作为数据源,通常采用较高阶的组件,而不是自己写,比如React Redux的connect(), Relay的createContainer(), Flux Utils的Container.create()

仅通过属性获取数据和回调

null

很少有自己的状态,即使有,也是自己的UI状态

null

除非他们需要的自己的状态,生命周期,或性能优化才会被写为功能组件

null

下面是一个可能会经常写的组件,评论列表组件,数据交互和展示都放到了一个组件里面。

// CommentList.js

class CommentList extends React.Component {

constructor() {

super();

this.state = { comments: [] }

}

componentDidMount() {

$.ajax({

url: "/my-comments.json",

dataType: 'json',

success: function(comments) {

this.setState({comments: comments});

}.bind(this)

});

}

render() {

return

    {this.state.comments.map(renderComment)}
;

}

renderComment({body, author}) {

return

  • {body}—{author}
  • ;

    }

    }

    我们对上面的组件进行拆分,把他拆分成容器组件 CommentListContainer.js 和展示组件 CommentList。

    // CommentListContainer.js

    class CommentListContainer extends React.Component {

    constructor() {

    super();

    this.state = { comments: [] }

    }

    componentDidMount() {

    $.ajax({

    url: "/my-comments.json",

    dataType: 'json',

    success: function(comments) {

    this.setState({comments: comments});

    }.bind(this)

    });

    }

    render() {

    return ;

    }

    }

    // CommentList.js

    class CommentList extends React.Component {

    constructor(props) {

    super(props);

    }

    render() {

    return

      {this.props.comments.map(renderComment)}
    ;

    }

    renderComment({body, author}) {

    return

  • {body}—{author}
  • ;

    }

    }

    优势:

    展示和容器更好的分离,更好的理解应用程序和UI

    重用性高,展示组件可以用于多个不同的state数据源

    展示组件就是你的调色板,可以把他们放到单独的页面,在不影响应用程序的情况下,让设计师调整UI

    迫使你分离标签,达到更高的可用性

    有状态组件和无状态组件

    下面是一个最简单的无状态组件的例子:

    function HelloComponent(props, /* context */) {

    return

    Hello {props.name}

    }

    ReactDOM.render(, mountNode)

    可以看到,原本需要写“类”定义(React.createClass 或者 class YourComponent extends React.Component)来创建自己组件的定义(有状态组件),现在被精简成了只写一个 render 函数。更值得一提的是,由于仅仅是一个无状态函数,React 在渲染的时候也省掉了将“组件类” 实例化的过程。

    结合 ES6 的解构赋值,可以让代码更精简。例如下面这个 Input 组件:

    function Input({ label, name, value, ...props }, { defaultTheme }) {

    const { theme, autoFocus, ...rootProps } = props

    return (

    htmlFor={name}

    children={label || defaultLabel}

    {...rootProps}

    >

    name={name}

    type="text"

    value={value || ''}

    theme={theme || defaultTheme}

    {...props}

    />

    )}

    Input.contextTypes = {defaultTheme: React.PropTypes.object};

    无状态组件不像上述两种方法在调用时会创建新实例,它创建时始终保持了一个实例,避免了不必要的检查和内存分配,做到了内部优化。

    无状态组件不支持 "ref"

    高阶组件

    高阶组件通过函数和闭包,改变已有组件的行为,本质上就是 Decorator 模式在 React 的一种实现。

    当写着写着无状态组件的时候,有一天忽然发现需要状态处理了,那么无需彻底返工:)

    往往我们需要状态的时候,这个需求是可以重用的。

    高阶组件加无状态组件,则大大增强了整个代码的可测试性和可维护性。同时不断“诱使”我们写出组合性更好的代码。

    高阶函数

    function welcome() {

    let username = localStorage.getItem('username');

    console.log('welcome ' + username);

    }

    function goodbey() {

    let username = localStorage.getItem('username');

    console.log('goodbey ' + username);

    }

    welcome();

    goodbey();

    我们发现两个函数有一句代码是一样的,这叫冗余唉。(平时可能会有一大段代码的冗余)。

    下面我们要写一个中间函数,读取username,他来负责把username传递给两个函数。

    function welcome(username) {

    console.log('welcome ' + username);

    }

    function goodbey(username) {

    console.log('goodbey ' + username);

    }

    function wrapWithUsername(wrappedFunc) {

    let newFunc = () => {

    let username = localStorage.getItem('username');

    wrappedFunc(username);

    };

    return newFunc;

    }

    welcome = wrapWithUsername(welcome);

    goodbey = wrapWithUsername(goodbey);

    welcome();

    goodbey();

    好了,我们里面的 wrapWithUsername 函数就是一个“高阶函数”。

    他做了什么?他帮我们处理了 username,传递给目标函数。我们调用最终的函数 welcome的时候,根本不用关心 username是怎么来的。

    举一反三的高阶组件

    下面是两个冗余的组件。

    import React, {Component} from 'react'

    class Welcome extends Component {

    constructor(props) {

    super(props);

    this.state = {

    username: ''

    }

    }

    componentWillMount() {

    let username = localStorage.getItem('username');

    this.setState({

    username: username

    })

    }

    render() {

    return (

    welcome {this.state.username}

    )

    }

    }

    export default Welcome;

    import React, {Component} from 'react'

    class Goodbye extends Component {

    constructor(props) {

    super(props);

    this.state = {

    username: ''

    }

    }

    componentWillMount() {

    let username = localStorage.getItem('username');

    this.setState({

    username: username

    })

    }

    render() {

    return (

    goodbye {this.state.username}

    )

    }

    }

    export default Goodbye;

    我们可以通过刚刚高阶函数的思想来创建一个中间组件,也就是我们说的高阶组件。

    import React, {Component} from 'react'

    export default (WrappedComponent) => {

    class NewComponent extends Component {

    constructor() {

    super();

    this.state = {

    username: ''

    }

    }

    componentWillMount() {

    let username = localStorage.getItem('username');

    this.setState({

    username: username

    })

    }

    render() {

    return

    }

    }

    return NewComponent

    }

    import React, {Component} from 'react';

    import wrapWithUsername from 'wrapWithUsername';

    class Welcome extends Component {

    render() {

    return (

    welcome {this.props.username}

    )

    }

    }

    Welcome = wrapWithUsername(Welcome);

    export default Welcome;

    import React, {Component} from 'react';

    import wrapWithUsername from 'wrapWithUsername';

    class Goodbye extends Component {

    render() {

    return (

    goodbye {this.props.username}

    )

    }

    }

    Goodbye = wrapWithUsername(Goodbye);

    export default Goodbye;

    看到没有,高阶组件就是把 username 通过 props 传递给目标组件了。目标组件只管从 props里面拿来用就好了。

    为了代码的复用性,我们应该尽量减少代码的冗余。

    提取共享的state,如果有两个组件都需要加载同样的数据,那么他们会有相同的 componentDidMount 函数。

    找出重复的代码,每个组件中constructor 和 componentDidMount都干着同样的事情,另外,在数据拉取时都会显示Loading... 文案,那么我们应该思考如何使用高阶组件来提取这些方法。

    迁移重复的代码到高阶组件

    包裹组件,并且使用props替换state

    尽可能地简化

    组件开发基本思想

    单功能原则

    使用react时,组件或容器的代码在根本上必须只负责一块UI功能。

    让组件保持简单

    如果组件根本不需要状态,那么就使用函数定义的无状态组件。

    从性能上来说,函数定义的无状态组件 > ES6 class 定义的组件 > 通过 React.createClass() 定义的组件。

    仅传递组件所需要的属性。只有当属性列表太长时,才使用{...this.props}进行传递。

    如果组件里面有太多的判断逻辑(if-else语句)通常意味着这个组件需要被拆分成更细的组件或模块。

    使用明确的命名能够让开发者明白它的功能,有助于组件复用。

    基本准则

    在shouldComponentUpdate中避免不必要的检查.

    尽量使用不可变数据类型(Immutable).

    编写针对产品环境的打包配置(Production Build).

    通过Chrome Timeline来记录组件所耗费的资源.

    在componentWillMount或者componentDidMount里面通过setTimeOut或者requestAnimationFram来延迟执行那些需要大量计算的任务.

    组件开发技巧

    form表单里的受控组件和不受控组件

    受控组件

    在大多数情况下,我们推荐使用受控组件来实现表单。在受控组件中,表单数据由 React 组件负责处理。下面是一个典型的受控组建。

    class NameForm extends React.Component {

    constructor(props) {

    super(props);

    this.state = {value: ''};

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

    this.handleSubmit = this.handleSubmit.bind(this);

    }

    handleChange(event) {

    this.setState({value: event.target.value});

    }

    handleSubmit(event) {

    alert('A name was submitted: ' + this.state.value);

    event.preventDefault();

    }

    render() {

    return (

    );

    }

    }

    设置表单元素的value属性之后,其显示值将由this.state.value决定,以满足React状态的同一数据理念。每次键盘敲击之后会执行handleChange方法以更新React状态,显示值也将随着用户的输入改变。

    对于受控组件来说,每一次 state(状态)变化都会伴有相关联的处理函数。这使得可以直接修改或验证用户的输入和提交表单。

    不受控组件

    因为不受控组件的数据来源是 DOM 元素,当使用不受控组件时很容易实现 React 代码与非 React 代码的集成。如果你希望的是快速开发、不要求代码质量,不受控组件可以一定程度上减少代码量。否则。你应该使用受控组件。

    一般情况下不受控组件我们使用ref来获取DOM元素进行操作。

    class NameForm extends React.Component {

    constructor(props) {

    super(props);

    this.handleSubmit = this.handleSubmit.bind(this);

    }

    handleSubmit(event) {

    alert('A name was submitted: ' + this.input.value);

    event.preventDefault();

    }

    render() {

    return (

    Name:

    this.input = input} />

    );

    }

    }

    组件条件判断

    三元函数组件判断渲染

    const sampleComponent = () => {

    return isTrue ?

    True!

    :

    false!

    };

    使用&&表达式替换不必要的三元函数

    const sampleComponent = () => {

    return isTrue ?

    True!

    :

    };

    const sampleComponent = () => {

    return isTrue &&

    True!

    };

    需要注意的是如果isTrue 为 0 ,其实会转换成 false,但是在页面中显示的时候,&&还是会返回0显示到页面中。

    多重嵌套判断

    // 问题代码

    const sampleComponent = () => {

    return (

    {flag && flag2 && !flag3

    ? flag4

    ?

    Blah

    : flag5

    ?

    Meh

    :

    Herp

    :

    Derp

    }

    )

    };

    解决方案:

    最佳方案: 将逻辑移到子组件内部

    使用IIFE(Immediately-Invoked Function Expression 立即执行函数)

    满足条件的时候使用return强制跳出函数

    const sampleComponent = () => {

    const basicCondition = flag && flag2 && !flag3;

    if (!basicCondition) return

    Derp

    ;

    if (flag4) return

    Blah

    ;

    if (flag5) return

    Meh

    ;

    return

    Herp

    }

    setState异步性

    在某些情况下,React框架出于性能优化考虑,可能会将多次state更新合并成一次更新。正因为如此,setState实际上是一个异步的函数。 如果在调用setState()函数之后尝试去访问this.state,你得到的可能还是setState()函数执行之前的结果。

    但是,有一些行为也会阻止React框架本身对于多次state更新的合并,从而让state的更新变得同步化。 比如: eventListeners, Ajax, setTimeout 等等。

    React框架之所以在选择在调用setState函数之后立即更新state而不是采用框架默认的方式,即合并多次state更新为一次更新,是因为这些函数调用(fetch,setTimeout等浏览器层面的API调用)并不处于React框架的上下文中,React没有办法对其进行控制。React在此时采用的策略就是及时更新,确保在这些函数执行之后的其他代码能拿到正确的数据(即更新过的state)。

    解决setState函数异步的办法?

    根据React官方文档,setState函数实际上接收两个参数,其中第二个参数类型是一个函数,作为setState函数执行后的回调。通过传入回调函数的方式,React可以保证传入的回调函数一定是在setState成功更新this.state之后再执行。

    this.setState({count: 1}, () => {

    console.log(this.state.count); // 1

    })

    React源码中setState的实现

    ReactComponent.prototype.setState = function(partialState, callback) {

    invariant(

    typeof partialState === 'object' ||

    typeof partialState === 'function' ||

    partialState == null,

    'setState(...): takes an object of state variables to update or a ' +

    'function which returns an object of state variables.'

    );

    this.updater.enqueueSetState(this, partialState);

    if (callback) {

    this.updater.enqueueCallback(this, callback, 'setState');

    }

    };

    updater的这两个方法,和React底层的Virtual Dom(虚拟DOM树)的diff算法有紧密的关系,所以真正决定同步还是异步的其实是Virtual DOM的diff算法。

    依赖注入

    在React中,想做依赖注入(Dependency Injection)其实相当简单。可以通过props来进行传递。但是,如果组件数量很多,并且组件嵌套层次很深的话,这种方式就不太合适。

    高阶组件

    // inject.jsx

    var title = 'React Dependency Injection';

    export default function inject(Component) {

    return class Injector extends React.Component {

    render() {

    return (

    {...this.state}

    {...this.props}

    title={ title }

    />

    )

    }

    };

    }

    // Title.jsx

    export default function Title(props) {

    return

    { props.title }

    ;

    }

    // Header.jsx

    import inject from './inject.jsx';

    import Title from './Title.jsx';

    var EnhancedTitle = inject(Title);

    export default function Header() {

    return (

    );

    }

    context

    React v16.3.0 之前的 Context:

    var context = { title: 'React in patterns' };

    class App extends React.Component {

    getChildContext() {

    return context;

    }

    // ...

    }

    App.childContextTypes = {

    title: PropTypes.string

    };

    class Inject extends React.Component {

    render() {

    var title = this.context.title;

    // ...

    }

    }

    Inject.contextTypes = {

    title: PropTypes.string

    };

    之前的 Context 作为一个实验性质的 API,直到 React v16.3.0 版本前都一直不被官方所提倡去使用,其主要原因就是因为在子组件中使用 Context 会破坏 React 应用的分型架构。

    这里的分形架构指的是从理想的 React 应用的根组件树中抽取的任意一部分都仍是一个可以直接运行的子组件树。在这个子组件树之上再包一层,就可以将它无缝地移植到任意一个其他的根组件树中。

    但如果根组件树中有任意一个组件使用了支持透传的 Context API,那么如果把包含了这个组件的子组件树单独拿出来,因为缺少了提供 Context 值的根组件树,这时的这个子组件树是无法直接运行的。

    并且他有一个致命缺陷:任何一个中间传递的组件shouldComponentUpdate 函数返回false,组件都不会得到更新。

    新的Context Api

    新的Context Api 采用声明式的写法,并且可以透过shouldComponentUpdate 函数返回false的组件继续向下传播,以保证目标组件一定可以接收到顶层组件 Context 值的更新,一举解决了现有 Context API 的两大弊端,也终于成为了 React 中的第一级(first-class) API。

    新的 Context API 分为三个组成部分:

    React.createContext 用于初始化一个 Context。

    XXXContext.Provider作为顶层组件接收一个名为 value的 prop,可以接收任意需要被放入 Context 中的字符串,数字,甚至是函数。

    XXXContext.Consumer作为目标组件可以出现在组件树的任意位置(在 Provider 之后),接收 children prop,这里的 children 必须是一个函数(context => ())用来接收从顶层传来的 Context。

    const ThemeContext = React.createContext('light');

    class App extends React.Component {

    render() {

    return (

    );

    }

    }

    function Toolbar(props) {

    return (

    );

    }

    function ThemedButton(props) {

    return (

    {theme =>

    );

    }

    _handleButtonClick() {

    console.log(`Button is clicked inside ${ this.state.name }`);

    // 将导致

    // Uncaught TypeError: Cannot read property 'state' of null

    }

    }

    我们可以通过下面三种方式简单实现this指向的绑定:

    在constructor 中事先绑定 this._buttonClick = this._handleButtonClick.bind(this);

    调用时使用箭头函数 this._buttonClick() }>

    ES7中的绑定操作符

    给setState传入回调函数

    setState() 不仅能接受一个对象,还能接受一个函数作为参数呢,该函数接受该组件前一刻的 state 以及当前的 props 作为参数,计算和返回下一刻的 state。

    // assuming this.state.count === 0

    this.setState({count: this.state.count + 1});

    this.setState({count: this.state.count + 1});

    this.setState({count: this.state.count + 1});

    // this.state.count === 1, not 3

    this.setState((prevState, props) => ({

    count: prevState.count + props.increment

    }));

    // Passing object

    this.setState({ expanded: !this.state.expanded });

    // Passing function

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

    组件切换技巧

    import HomePage from './HomePage.jsx';

    import AboutPage from './AboutPage.jsx';

    import UserPage from './UserPage.jsx';

    import FourOhFourPage from './FourOhFourPage.jsx';

    const PAGES = {

    home: HomePage,

    about: AboutPage,

    user: UserPage

    };

    const Page = (props) => {

    const Handler = PAGES[props.page] || FourOhFourPage;

    return

    };

    React style

    组件分类

    基础组件, 布局组件, 排版组件

    给无状态的纯UI组件应用样式

    请保持样式远离那些离不开state的组件. 比如路由, 视图, 容器, 表单, 布局等等不应该有任何的样式或者css class出现在组件上. 相反, 这些复杂的业务组件应该有一些带有基本功能的无状态UI组件组成.

    class SampleComponent extends Component {

    render() {

    return (

    name='username'

    value={username}

    onChange={this.handleChange}/>

    type='password'

    name='password'

    value={password}

    onChange={this.handleChange}/>

    type='submit'

    children='Sign In'/>

    )

    }

    }

    // 表达组件(带样式)

    const Button = ({

    ...props

    }) => {

    const sx = {

    fontFamily: 'inherit',

    fontSize: 'inherit',

    fontWeight: 'bold',

    textDecoration: 'none',

    display: 'inline-block',

    margin: 0,

    paddingTop: 8,

    paddingBottom: 8,

    paddingLeft: 16,

    paddingRight: 16,

    border: 0,

    color: 'white',

    backgroundColor: 'blue',

    WebkitAppearance: 'none',

    MozAppearance: 'none'

    }

    return (

    你可能感兴趣的:(react组件设计原则)