React 深刻系列2:组件分类

文:徐超,《React进阶之路》做者前端

受权发布,转载请注明做者及出处react


React 深刻系列2:组件分类

React 深刻系列,深刻讲解了React中的重点概念、特性和模式等,旨在帮助你们加深对React的理解,以及在项目中更加灵活地使用React。json

React 组件有不少种分类方式,常见的分类方式有函数组件和类组件,无状态组件和有状态组件,展现型组件和容器型组件。好吧,这又是一篇咬文嚼字的文章。可是,真正把这几组概念咬清楚、嚼明白后,对于页面的组件划分、组件之间的解耦是大有裨益的。api

函数组件和类组件

函数组件(Functional Component )和类组件(Class Component),划分依据是根据组件的定义方式。函数组件使用函数定义组件,类组件使用ES6 class定义组件。下面是函数组件和类组件的简单示例:数组

// 函数组件
function Welcome(props) {
  return <h1>Hello, {props.name}</h1>;
}

// 类组件
class Welcome extends React.Component {
  render() {
    return <h1>Hello, {this.props.name}</h1>;
  }
}

上面的两种写法是等价的,但函数组件的写法要比类组件简洁,不过类组件比函数组件功能更增强大。类组件能够维护自身的状态变量,即组件的state,类组件还有不一样的生命周期方法,可让开发者可以在组件的不一样阶段(挂载、更新、卸载),对组件作更多的控制。前端工程师

类组件有这么多优势,是否是咱们在开发中应该首选使用类组件呢?其实否则。函数组件更加专一和单一,承担的职责也更加清晰,它只是一个返回React 元素的函数,只关注对应UI的展示。函数组件接收外部传入的props,返回对应UI的DOM描述,仅此而已。固然,如上面例子所示,使用只包含一个render方法的类组件,能够实现和函数组件相同的效果。但函数组件的使用能够从思想上迫使你在设计组件时多作思考,更加关注逻辑和显示的分离,设计出更加合理的页面上组件树的结构。实际操做上,当一个组件不须要管理自身状态时,能够把它设计成函数组件,当你有足够的理由发现它须要“升级”为类组件时,再把它改造为类组件。由于函数组件“升级”为类组件是有必定成本的,这样就会要求你作这个改造前更认真地思考其合理性,而不是仅仅为了一时的方便就使用类组件。less

无状态组件和有状态组件

无状态组件(Stateless Component )和有状态组件(Stateful Component),划分依据是根据组件内部是否维护state。无状态组件内部不使用state,只根据外部组件传入的props返回待渲染的React 元素。有状态组件内部使用state,维护自身状态的变化,有状态组件根据外部组件传入的props和自身的state,共同决定最终返回的React 元素。函数

很容易知道,函数组件必定是无状态组件,类组件则既能够充当无状态组件,也能够充当有状态组件。但如上文所述,当一个组件不须要管理自身状态时,也就是无状态组件,应该优先设计为函数组件。fetch

展现型组件和容器型组件

展现型组件(Presentational Component)和容器型组件(Container Component),划分依据是根据组件的职责。this

展现型组件的职责是:组件UI长成什么样。展现型组件不关心组件使用的数据是如何获取的,以及组件数据应该如何修改,它只须要知道有了这些数据后,组件UI是什么样子的便可。外部组件经过props传递给展现型组件所需的数据和修改这些数据的回调函数,展现型组件只是它们的使用者。展现型组件通常是无状态组件,不须要state,由于展现型组件不须要管理数据,但当展现型组件须要管理自身的UI状态时,例如控制组件内部弹框的显示与隐藏,是可使用state的,这时的state属于UI state。既然大部分状况下展现型组件不须要state,应该优先考虑使用函数组件实现展现型组件。

容器型组件的职责是:组件数据如何工做。容器型组件须要知道如何获取子组件所需数据,以及这些数据的处理逻辑,并把数据和逻辑经过props提供给子组件使用。容器型组件通常是有状态组件,由于它们须要管理页面所需数据。

例如,下面的例子中,UserListContainer是一个容器型组件,它获取用户列表数据,而后把用户列表数据传递给展现型组件UserList,由UserList负责UI的展示。

class UserListContainer extends React.Component{
  constructor(props){
    super(props);
    this.state = {
      users: []
    }
  }
  
  componentDidMount() {
    var that = this;
    fetch('/path/to/user-api').then(function(response) {
      response.json().then(function(data) {
        that.setState({users: data})
      });
    });
  }

  render() {
    return (
      <UserList users={this.state.users} />
    )
  }
}

function UserList(props) {
  return (
    <div>
      <ul className="user-list">
        {props.users.map(function(user) {
          return (
            <li key={user.id}>
              <span>{user.name}</span>
            </li>
          );
        })}
      </ul>
    </div>
  )  
}

展现型组件和容器型组件是能够互相嵌套的,展现型组件的子组件既能够包含展现型组件,也能够包含容器型组件,容器型组件也是如此。例如,当一个容器型组件承担的数据管理工做过于复杂时,能够在它的子组件中定义新的容器型组件,由新组件分担数据的管理。展现型组件和容器型组件的划分彻底取决于组件所作的事情。

总结

经过上面的介绍,能够发现这三组概念有不少重叠部分。这三组概念都体现了关注点分离的思想:UI展示和数据逻辑的分离。函数组件、无状态组件和展现型组件主要关注UI展示,类组件、有状态组件和容器型组件主要关注数据逻辑。但因为它们的划分依据不一样,它们并不是彻底等价的概念。它们之间的关联关系能够概括为:函数组件必定是无状态组件,展现型组件通常是无状态组件;类组件既能够是有状态组件,又能够是无状态组件,容器型组件通常是有状态组件。

下篇预告:

React 深刻系列3:State 和 Props


新书推荐《React进阶之路》

做者:徐超

毕业于浙江大学,硕士,资深前端工程师,长期就任于能源物联网公司远景智能。8年软件开发经验,熟悉大前端技术,拥有丰富的Web前端和移动端开发经验,尤为对React技术栈和移动Hybrid开发技术有深刻的理解和实践经验。



美团点评广告平台大前端团队招收2019\2020年前端实习生

有意者邮件:yao.zhou@meituan.com

相关文章
相关标签/搜索