React高阶组件--render props、高阶组件(React学习笔记_06)

React - render props和高阶组件

1,render props模式

使用步骤

  • 1,创建一个组件,在组件中提供复用的状态逻辑代码
  • 2,将要复用的状态作为props.render(state)方法的参数,暴露到组件外部
  • 3,使用props.render()的返回值作为要渲染的内容
class 组件名 extends React.Component {
  state = {}
  render() {
    return this.props.render(this.state)
  }
}

<组件名 render={i => {
  return (
    <div>{i.xx}</div>
  )
}}/>

示例:

class Mouse extends React.Component {
  state = {
    x: 0,
    y: 0
  }

  handleMouseMove = e => {
    this.setState({
      x: e.clientX,
      y: e.clientY
    })
  }
  // 监听鼠标移动事件
  componentDidMount() {
    window.addEventListener('mousemove', this.handleMouseMove)
  }
  render() {
    return this.props.render(this.state)
  }
}
class App extends React.Component {
  render() {
    return (
      <div>
        <h1>哈哈哈哈</h1>
        <Mouse render={i => {
          return (
            <div>鼠标位置:{i.x}, {i.y}</div>
          )
        }}/>

        <Mouse render={i => {
          return (
            <img src={img} style={{position: 'absolute', top: i.y - 100, left: i.x - 100}}></img>
          )
        }}/>
      </div>
    )
  }
}

children代替render属性

  • 并不是该模式叫render props就必须使用名为render的prop,实际上可以使用任意名称的prop
  • 把prop是一个函数并且告诉组件要渲染什么内容的技术叫做:render props模式
  • 推荐使用children代替render属性
class Mouse extends React.Component {
  state = {
    x: 0,
    y: 0
  }

  handleMouseMove = e => {
    this.setState({
      x: e.clientX,
      y: e.clientY
    })
  }
  // 监听鼠标移动事件
  componentDidMount() {
    window.addEventListener('mousemove', this.handleMouseMove)
  }

  // 代码优化项 2 
  // 在卸载的时候解除mousemove事件绑定
  componentWillUnmount() {
    window.removeEventListener('mousemove', this.handleMouseMove)
  }
  render() {
    // return this.props.render(this.state)
    // children代替render 
    return this.props.children(this.state)
  }
}

// 代码优化项 1
Mouse.prototypes = {
  children: PropTypes.func.isRequired,
}
class App extends React.Component {
  render() {
    return (
      <div>
        <h1>哈哈哈哈</h1>
        {/* <Mouse render={i => {
          return (
            <div>鼠标位置:{i.x}, {i.y}</div>
          )
        }}/> */}

        {/* children代替render */}
        <Mouse>
          {
            ({x, y}) => {
              return (
                <div>鼠标位置:{x}, {y}</div>
              )
            }
          }
        </Mouse>

        {/* <Mouse render={i => {
          return (
            <img src={img} style={{position: 'absolute', top: i.y - 100, left: i.x - 100}}></img>
          )
        }}/> */}

        {/* children代替render  */}
        <Mouse>
          {
            ({x, y}) => {
              return (
                <img src={img} style={{position: 'absolute', top: y - 100, left: x - 100}}></img>
              )
            }
          }
        </Mouse>
      </div>
    )
  }
}

2, 高阶组件(HOC)

目的是实现状态逻辑复用。

使用步骤:

  • 1,创建一个函数,名称约定以with开头
  • 2,指定函数参数,参数应该以大写字母开头(作为要渲染的组件)
  • 3,在函数内部创建一个类组件,提供复用的状态逻辑代码,并返回
  • 4,在该组件中,渲染参数组件,同时将状态通过prop传递给参数组件
  • 5,调用该高阶组件,传入要增强的组件,通过返回值拿到增强后的组件,并将其渲染到页面中
function withMouse(WrappendComponent) {
  class Mouse extends React.Component{
    return Mouse
  }
}

// Mouse组件的render方法中:
return <WrappendComponent {...this.state}/>

// 创建组件
const MousePosition = widthMouse(Position)

// 渲染组件
<MousePosition />

以上示例改为高阶组件:

// 创建高阶组件
function widthMouse(WrappedComponent) {
  // 该组件提供复用的状态逻辑
  class Mouse extends React.Component {
    state = {
      x: 0,
      y: 0
    }

    handleMouseMove = e => {
      this.setState({
        x: e.clientX,
        y: e.clientY
      })
    }
    componentDidMount() {
      window.addEventListener('mousemove', this.handleMouseMove)
    }

    componentWillUnmount() {
      window.removeEventListener('mousemove', this.handleMouseMove)
    }

    render() {
      return <WrappedComponent {...this.state}/>
    }
  }

  return Mouse
}

// 用来测试高阶组件 获取鼠标移动的位置的组件
const Position = props => (
  <p>
    鼠标当前位置: (x: {props.x}, y: {props.y})
  </p>
)

// 用来测试获取图片跟随鼠标动的组件
const ImgPosition = props => (
  <img src={img} style={{
    position: 'absolute',
    top: props.y - 100,
    left: props.x - 100
  }}></img>
)

// 获取增强后的组件
const MousePosition = widthMouse(Position)

const MouseImg = widthMouse(ImgPosition)

class App extends React.Component {
  render() {
    return (
      <div>
        <h1>高阶组件</h1>
        <MousePosition/>
        <MouseImg/>
      </div>
    )
  }
}
设置displayName
  • 使用高阶组件存在的问题:得到的两个组件名称相同
  • 原因:默认情况下,React使用组件名称作为displayName
  • 解决方式: 为高阶组件设置displayName便于调试的时候区分不同的组件
  • displayName的作用:用于设置调试信息(React developer tools)
  • 设置方式
Mouse.displayName = `widthMouse${getDisplayName(WrappedComponent)}`

function getDisplayName(WrappedComponent) {
  return WrappedComponent.displayName || WrappedComponent.name || 'xiaowang'
}

以上示例改写为设置displayName

function widthMouse(WrappedComponent) {
  // 该组件提供复用的状态逻辑
  class Mouse extends React.Component {
    state = {
      x: 0,
      y: 0
    }

    handleMouseMove = e => {
      this.setState({
        x: e.clientX,
        y: e.clientY
      })
    }
    componentDidMount() {
      window.addEventListener('mousemove', this.handleMouseMove)
    }

    componentWillUnmount() {
      window.removeEventListener('mousemove', this.handleMouseMove)
    }

    render() {
      return <WrappedComponent {...this.state}/>
    }
  }
  // 设置displayName
  Mouse.displayName = `widthMouse${getDisplayName(WrappedComponent)}`  // widthMousePosition widthMouseImgPosition

  return Mouse
}

function getDisplayName(WrappedComponent) {
  console.log('WrappedComponent=====>', WrappedComponent.displayName)  // undefined undefined
  console.log('WrappedComponent=====>', WrappedComponent.name)  // Position ImgPosition
  return WrappedComponent.displayName || WrappedComponent.name || 'xiaoWangComponent'
}

// 用来测试高阶组件 获取鼠标移动的位置的组件
const Position = props => (
  <p>
    鼠标当前位置: (x: {props.x}, y: {props.y})
  </p>
)

// 用来测试获取图片跟随鼠标动的组件
const ImgPosition = props => (
  <img src={img} style={{
    position: 'absolute',
    top: props.y - 100,
    left: props.x - 100
  }}></img>
)

// 获取增强后的组件
const MousePosition = widthMouse(Position)

const MouseImg = widthMouse(ImgPosition)

class App extends React.Component {
  render() {
    return (
      <div>
        <h1>高阶组件</h1>
        <MousePosition/>
        <MouseImg/>
      </div>
    )
  }
}
高阶组件 传递props
  • props丢失
  • 原因:高阶组件没有往下传递props
  • 解决方式:渲染WrappedComponent时,将state和this.props一起传递给组件
  • 传递方式
<WrappedComponent {...this.state} {...this.props}></WrappedComponent>

以上示例涉及到修改的代码:

function widthMouse(WrappedComponent) {
  // 该组件提供复用的状态逻辑
  // class Mouse extends React.Component {
  //   state = {
  //     x: 0,
  //     y: 0
  //   }

    // handleMouseMove = e => {
    //   this.setState({
    //     x: e.clientX,
    //     y: e.clientY
    //   })
    // }
    // componentDidMount() {
    //   window.addEventListener('mousemove', this.handleMouseMove)
    // }

    // componentWillUnmount() {
    //   window.removeEventListener('mousemove', this.handleMouseMove)
    // }

    render() {
      // Mouse组件中可以拿到other这个属性
      console.log('this.props=====>', this.props)
      return <WrappedComponent {...this.state} {...this.props}/>
    }
  }
  // 设置displayName
  // Mouse.displayName = `widthMouse${getDisplayName(WrappedComponent)}`  // widthMousePosition widthMouseImgPosition

  return Mouse
}

// function getDisplayName(WrappedComponent) {
//   console.log('WrappedComponent=====>', WrappedComponent.displayName)  // undefined undefined
//   console.log('WrappedComponent=====>', WrappedComponent.name)  // Position ImgPosition
//   return WrappedComponent.displayName || WrappedComponent.name || 'xiaoWangComponent'
// }

// 用来测试高阶组件 获取鼠标移动的位置的组件
const Position = props => {
  // position组件没有拿到other属性
  console.log('position组件:=====>', props)
  return (
    <p>
      鼠标当前位置: (x: {props.x}, y: {props.y})
    </p>
  )
}

// 用来测试获取图片跟随鼠标动的组件
// const ImgPosition = props => (
//   <img src={img} style={{
//     position: 'absolute',
//     top: props.y - 100,
//     left: props.x - 100
//   }}></img>
// )

// 获取增强后的组件
// const MousePosition = widthMouse(Position)

// const MouseImg = widthMouse(ImgPosition)

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

推荐阅读更多精彩内容