Render Props - New pattern in React

在软件开发的过程中Code Reuse和可读性一直是开发人员致力于解决的问题,在React社区中,以往出现了很多的Pattern来解决这个问题,例如Mixin,HOC等等。Render Props是React社区中里提出的另外的一种Pattern。由React Router的Co-author Michael Jackson 提出,它与HOC等有什么区别呢?又解决了什么问题呢?在这篇文章来一探究竟。

问题的引出

所有的Pattern或者solution都是为了解决问题而提出的,那么render props到底是为了解决什么问题呢?我们先来看一个例子。例如在我们的App里我们实现了这样一个component

import React from 'react';

class App extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      x: 0,
      y: 0
    };
    this.handleMouseMove = this.handleMouseMove.bind(this);
  }

  handleMouseMove(event) {
    this.setState({
      x: event.clientX,
      y: event.clientY
    });
  }

  render() {
    return (
      <div style={{ height: 800 }} onMouseMove={this.handleMouseMove}>
        <p>
          The mouse is at ({this.state.x}, {this.state.y})
        </p>
      </div>
    );
  }
}    

Code Sandbox

熟悉React的同学都知道这是一个很简单的App,他可以追踪鼠标位置给到自己。但是问题来了,万一哪天你的朋友看到这个component,说这个feature很Cool,他的app里能不能用上呢?那么该怎么解决这个问题呢?

Code Repeat

首先第一种想法很直接,可以把这个component做为一个父Component,另外一个Component作为一个子Component。来看代码。

import React from 'react';

const Dog = ({mouse}) => (
    <div>{mouse.x}, {mouse.y}</div>
)

class MouseDog extends React.Component {
    constructor(props) {
    super(props);
    this.state = {
      x: 0,
      y: 0
    };
    this.handleMouseMove = this.handleMouseMove.bind(this);
  }

  handleMouseMove(event) {
    this.setState({
      x: event.clientX,
      y: event.clientY
    });
  }

  render() {
    return (
      <div style={{ height: 800 }} onMouseMove={this.handleMouseMove}>
        <p>
          The mouse is at ({this.state.x}, {this.state.y})
          <Dog mouse={mouse} />
        </p>
      </div>
    );
  }
}

现在我们有了两个Component, Dog 和 DogWithMouse,其实Dog作为DogWithMouse的子Component,那么DogWithMouse Component就能利用鼠标位置。好像这样是把问题解决了,但是如果我们有另外一个Component也想要trace mouse这个feature怎么办呢?难道再重新这样写一个Component吗?显然这种方式是不可取的。

HOC

如果熟悉react的同学,可能会想到,这很简单呀,我们写一个High order Component(HOC)就行了,的确HOC可以解决这个问题,我们来看看HOC的代码是怎么样的。


import React from 'react';

const withMouse = (Component) => {
  return class extends React.Component {
    constructor(props) {
      super(props);
      this.state = {
        x: 0,
        y: 0
      };
      this.handleMouseMove = this.handleMouseMove.bind(this);
    }

    handleMouseMove(event) {
      this.setState({
        x: event.clientX,
        y: event.clientY
      });
    }

    render() {
      return (
        <div style={{ height: 800 }} onMouseMove={this.handleMouseMove}>
          <Component mouse={this.state} {...this.props} />
        </div>
      );
    }
  }
}
const Dog = ({ mouse }) => <div>{mouse.x}, {mouse.y}</div>

const EnhancedDog = withMouse(Dog)

在上面的代码里,我们创建了一个withMouse的HOC, 这样就把这部分逻辑抽象了出来。其他的Component想要使用我们mouse tracking的feature。只要将Component传入HOC就能得到一个enhance过的Component。pretty cool, right?如果对于HOC不太了解,可以参考react 的官方文档进行了解。 HOC

HOC似乎已经完美的解决了问题,但是这种方式有没有什么缺点呢?似乎是有下面的一些缺点。

  1. 间接的props传入。
    让我们重新看一看Dog Component,他的props传入了mouse,但是在使用这个Component的时候,并没有地方传入了mouse。这是因为mouse是在HOC里传入的。你大概会纳闷这个Mouse是从那来的。如果维护一个大项目的时候,有时候你就会发现这种间接的props传入,是有多么坑了。

  2. 命名冲突的问题。
    用过react的同学不知道有没有经历过大量使用HOC的场景,笔者是见过多层HOC嵌套使用的项目代码。就像这样: const NewComponent = withA(withB(withC(...(Component))))
    如果你用过这样的多层嵌套的HOC,那么就可能会发生命名冲突的问题。来看代码:


const withB = (Component) => {
  return class extends React.Component {
    render() {
      return (
        <div style={{ height: 800 }} >
          <Component mouse={'hehe'} {...this.props} />
        </div>
      );
    }
  }
}

const EnhancedDog = withB(withMouse(Dog))

在这里定义另外的一个HOC withB。它也给props上传了一个同样的mouse, 如果再用它来wrap withMouse(Dog) 时,你会发现mouse tracking的feature就不work,但是React不会有任何提示,这是命名冲突的问题。如果你有一个七八层的wrapp到一起的Component,如果出现这样的命名冲突的问题,那么就有的debug了。

Render props

好了,既然HOC有这样的问题,那有没有什么方式既可以做到code reuse,同时也可以避免HOC的这些问题呢?答案就是Render Props。Render Props是React Traning团队提出的一种新的React中组织Component的方式,同时也是十分的简单,我们来看上边的例子中,如果用render props如何解决code reuse的问题。

import React from 'react';
import { render } from "react-dom";

class Mouse extends React.Component {
    constructor(props) {
    super(props);
    this.state = {
      x: 0,
      y: 0
    };
    this.handleMouseMove = this.handleMouseMove.bind(this);
  }

  handleMouseMove(event) {
    this.setState({
      x: event.clientX,
      y: event.clientY
    });
  }

  render() {
    return (
      <div style={{ height: 800 }} onMouseMove={this.handleMouseMove}>
        {this.props.render(this.state)}
      </div>
    );
  }
}

const DogMouse = () => (
  <Mouse render={({x, y}) => (<div>this mouose is ({x}, {y})</div>})>
)

render(<DogMouse/>, document.getElementById('app'))

CodeSandBox

上边就是render props的一个简单例子,我们看完会发现特别的简单,没什么特别的。Mouse就是一个普通的Component,唯一不同的是这个Component上定义了一个叫render的prop,是一个函数。在Mouse中会将state作为参数调用这个函数。而创建的DogMouse同样也是一个普通的Component,其中定义render的方法,用于定义要render什么Component。我们看看render props和HOC相比解决了什么问题:

  1. code reuse。和HOC一样的,render props同样解决了code reuse的问题。把mouse tracking的feature抽象成为一个Mouse的Component,如果有另外一个Component像复用这个feature的时候,简单的定义另外一个Component定义相应的render props就好了。

  2. 间接的props传入,render props 没有这个问题,可以清楚的看到props(x, y)是从什么地方出入的.

  3. 命名冲突。利用render props是没有Component wrapp的,所以除非定义Component时候自己命名重复,否则不会有命名冲突的问题。

从上面的例子上看,render props可以再大多数情况下替代HOC。react的官方文档上,目前也正式的介绍了render props,下次让你想用HOC的时候,来试一试Render props把。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容