• 组合 vs 继承
    • 包含关系
    • 特例关系
    • 那么继承呢?

    组合 vs 继承

    React 有十分强大的组合模式。我们推荐使用组合而非继承来实现组件间的代码重用。

    在这篇文档中,我们将考虑初学 React 的开发人员使用继承时经常会遇到的一些问题,并展示如何通过组合思想来解决这些问题。

    包含关系

    有些组件无法提前知晓它们子组件的具体内容。在 Sidebar(侧边栏)和 Dialog(对话框)等展现通用容器(box)的组件中特别容易遇到这种情况。

    我们建议这些组件使用一个特殊的 children prop 来将他们的子组件传递到渲染结果中:

    1. function FancyBorder(props) {
    2. return (
    3. <div className={'FancyBorder FancyBorder-' + props.color}>
    4. {props.children}
    5. </div>
    6. );
    7. }

    这使得别的组件可以通过 JSX 嵌套,将任意组件作为子组件传递给它们。

    1. function WelcomeDialog() {
    2. return (
    3. <FancyBorder color="blue">
    4. <h1 className="Dialog-title">
    5. Welcome
    6. </h1>
    7. <p className="Dialog-message">
    8. Thank you for visiting our spacecraft!
    9. </p>
    10. </FancyBorder>
    11. );
    12. }

    在 CodePen 上尝试

    <FancyBorder> JSX 标签中的所有内容都会作为一个 children prop 传递给 FancyBorder 组件。因为 FancyBorder{props.children} 渲染在一个 <div> 中,被传递的这些子组件最终都会出现在输出结果中。

    少数情况下,你可能需要在一个组件中预留出几个“洞”。这种情况下,我们可以不使用 children,而是自行约定:将所需内容传入 props,并使用相应的 prop。

    1. function SplitPane(props) {
    2. return (
    3. <div className="SplitPane">
    4. <div className="SplitPane-left">
    5. {props.left}
    6. </div>
    7. <div className="SplitPane-right">
    8. {props.right}
    9. </div>
    10. </div>
    11. );
    12. }
    13. function App() {
    14. return (
    15. <SplitPane
    16. left={
    17. <Contacts />
    18. }
    19. right={
    20. <Chat />
    21. } />
    22. );
    23. }

    在 CodePen 上尝试

    <Contacts /><Chat /> 之类的 React 元素本质就是对象(object),所以你可以把它们当作 props,像其他数据一样传递。这种方法可能使你想起别的库中“槽”(slot)的概念,但在 React 中没有“槽”这一概念的限制,你可以将任何东西作为 props 进行传递。

    特例关系

    有些时候,我们会把一些组件看作是其他组件的特殊实例,比如 WelcomeDialog 可以说是 Dialog 的特殊实例。

    在 React 中,我们也可以通过组合来实现这一点。“特殊”组件可以通过 props 定制并渲染“一般”组件:

    1. function Dialog(props) {
    2. return (
    3. <FancyBorder color="blue">
    4. <h1 className="Dialog-title">
    5. {props.title}
    6. </h1>
    7. <p className="Dialog-message">
    8. {props.message}
    9. </p>
    10. </FancyBorder>
    11. );
    12. }
    13. function WelcomeDialog() {
    14. return (
    15. <Dialog
    16. title="Welcome"
    17. message="Thank you for visiting our spacecraft!" />
    18. );
    19. }

    在 CodePen 上尝试

    组合也同样适用于以 class 形式定义的组件。

    1. function Dialog(props) {
    2. return (
    3. <FancyBorder color="blue">
    4. <h1 className="Dialog-title">
    5. {props.title}
    6. </h1>
    7. <p className="Dialog-message">
    8. {props.message}
    9. </p>
    10. {props.children}
    11. </FancyBorder>
    12. );
    13. }
    14. class SignUpDialog extends React.Component {
    15. constructor(props) {
    16. super(props);
    17. this.handleChange = this.handleChange.bind(this);
    18. this.handleSignUp = this.handleSignUp.bind(this);
    19. this.state = {login: ''};
    20. }
    21. render() {
    22. return (
    23. <Dialog title="Mars Exploration Program"
    24. message="How should we refer to you?">
    25. <input value={this.state.login}
    26. onChange={this.handleChange} />
    27. <button onClick={this.handleSignUp}>
    28. Sign Me Up!
    29. </button>
    30. </Dialog>
    31. );
    32. }
    33. handleChange(e) {
    34. this.setState({login: e.target.value});
    35. }
    36. handleSignUp() {
    37. alert(`Welcome aboard, ${this.state.login}!`);
    38. }
    39. }

    在 CodePen 上尝试

    那么继承呢?

    在 Facebook,我们在成百上千个组件中使用 React。我们并没有发现需要使用继承来构建组件层次的情况。

    Props 和组合为你提供了清晰而安全地定制组件外观和行为的灵活方式。注意:组件可以接受任意 props,包括基本数据类型,React 元素以及函数。

    如果你想要在组件间复用非 UI 的功能,我们建议将其提取为一个单独的 JavaScript 模块,如函数、对象或者类。组件可以直接引入(import)而无需通过 extend 继承它们。