组合vs继承

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

props.children

有些组件无法提前知晓它们子组件的具体内容。在 Sidebar(侧边栏)和 Dialog(对话框)等展现通用容器(box)的组件中特别容易遇到这种情况。
我们建议这些组件使用一个特殊的 children prop 来将他们的子组件传递到渲染结果中:

  1. class SideBarItems extends React.Component {
  2. render(){
  3. return (
  4. <ul>
  5. <li>首页</li>
  6. <li>用户管理</li>
  7. <li>商品管理</li>
  8. <li>内容管理</li>
  9. </ul>
  10. )
  11. }
  12. }
  13. class SideBar extends React.Component {
  14. constructor(props){
  15. super(props);
  16. }
  17. render(){
  18. return (
  19. <div>
  20. {this.props.children}
  21. </div>
  22. )
  23. }
  24. }
  25. class Content extends React.Component {
  26. render(){
  27. return (
  28. <div>
  29. 这是内容部分
  30. </div>
  31. )
  32. }
  33. }
  34. class FootBar extends React.Component {
  35. render(){
  36. return (
  37. <div>
  38. 这是底部部分
  39. </div>
  40. )
  41. }
  42. }
  43. class BaseLayout extends React.Component {
  44. constructor(props){
  45. super(props);
  46. }
  47. render(){
  48. return (
  49. <div>
  50. {/*props.children是默认熟悉,指当前组件标签的内容部分*/}
  51. {this.props.children}
  52. </div>
  53. )
  54. }
  55. }
  56. class HelloWorld extends React.Component {
  57. render(){
  58. return (
  59. <div>
  60. {/*类似于vue的插槽slot */}
  61. <BaseLayout>
  62. <SideBar>
  63. <SideBarItems/>
  64. </SideBar>
  65. <Content/>
  66. <FootBar/>
  67. </BaseLayout>
  68. </div>
  69. )
  70. }
  71. }
  72. // 通过调用React自身方法render可以得到当前组件的实例对象,并渲染到页面容器.
  73. ReactDOM.render(<App />, document.getElementById("root"));

image.pngimage.png

具名插槽

少数情况下,你可能需要在一个组件中预留出几个“洞”。这种情况下,我们可以不使用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. }

特例关系

  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. }