image.png

组件生命周期

每一个组件都有几个你可以重写以让代码在处理环节的特定时期运行的“生命周期方法”。
方法中带有前缀 will 的在特定环节之前被调用,
而带有前缀 did 的方法则会在特定环节之后被调用。

装载(Mounting)

这些方法会在组件实例被创建和插入 DOM 中时被调用:

  • constructor()
  • componentWillMount()
  • render()
  • componentDidMount()

    更新(Updating)

    属性或状态的改变会触发一次更新。当一个组件在被重新渲染时,这些方法将会被调用:

  • componentWillReceiveProps()

  • shouldComponentUpdate()
  • componentWillUpdate()
  • render()
  • componentDidUpdate()

    卸载(Unmounting)

    当一个组件被从 DOM 中移除时,该方法被调用:

  • componentWillUnmount()

    其他 API

    每一个组件还提供了其他的 API:

  • setState()

  • forceUpdate()

    类属性

  • defaultProps

    实例属性

  • props

  • state
  • config(小程序专属)

    参考

    render()

    render() 方法是必须的。
    当被调用时,render 方法必须返回一个 Taro 组件
    (可以是内置组件也可以是自定义组件)或一个 falsy 的值。

render() 函数应该纯净,
意味着其不应该改变组件的状态,
其每次调用都应返回相同的结果,同时不直接和浏览器/小程序交互。

若需要和浏览器/小程序交互,将任务放在componentDidMount() 阶段或其他的生命周期方法。

保持 render() 方法纯净使得组件更容易思考。

在 React/Nerv 中,render() 可以返回多种数据结构,但 Taro 暂时只支持两种。因为 Taro 必须把 JSX 编译成微信小程序模板。当 return 的值为 falsy 时,实际上会编译成小程序的 wx:if 标签。

constructor()

  1. constructor(props)

Taro 组件的构造函数将会在装配之前被调用。

当为一个 Taro.Component 子类定义构造函数时,
你应该在任何其他的表达式之前调用 super(props)
否则,this.props 在构造函数中将是未定义,并可能引发异常。

构造函数是初始化状态的合适位置。
若你不初始化状态且不绑定方法,那你也不需要为你的 Taro 组件定义一个构造函数。

在 Taro 中,即便你不写 constructor(),编译到微信小程序时也会自动给你加上,因为 Taro 运行时框架需要在 constructor() 中多做一些事情。

可以基于属性来初始化状态。这样有效地“分离(forks)”属性并根据初始属性设置状态。

static getDerivedStateFromProps()

自 v1.3.0 起可用

  1. static getDerivedStateFromProps(props, state)

<br />
getDerivedStateFromProps 会在调用 render 方法之前调用,
并且在初始挂载及后续更新时都会被调用。

它应返回一个对象来更新 state,如果返回 null 则不更新任何内容。

此方法无权访问组件实例(this)。

如果你需要,可以通过提取组件 props 的纯函数及 class 之外的状态,
getDerivedStateFromProps() 和其他 class 方法之间重用代码。

componentWillReceiveProps 不同,
getDerivedStateFromProps 会在每次 render 前触发此方法
componentWillReceiveProps 会在父组件重新渲染时调用。

  1. export default class ButtonSelectable extends Component {
  2. static propTypes = {
  3. selected: PropTypes.bool,
  4. onClick: PropTypes.func
  5. };
  6. constructor(props) {
  7. super(props);
  8. this.state = {
  9. isSelected: props.selected
  10. };
  11. }
  12. static getDerivedStateFromProps(props, state) {
  13. if (props.selected !== state.isSelected) {
  14. return {
  15. isSelected: props.selected
  16. };
  17. }
  18. return null;
  19. }
  20. handleClick = event => {
  21. this.setState({
  22. isSelected: !this.state.isSelected
  23. });
  24. };
  25. render() {
  26. return (
  27. <Button
  28. className={`B-selectable ${
  29. this.state.isSelected ? "button-selectable-selected" : ""
  30. }`}
  31. onClick={this.handleClick}
  32. >
  33. {this.state.isSelected ? "Selected!" : "Not selected..."}
  34. </Button>
  35. );
  36. }
  37. }

请注意: getDerivedStateFromProps() 如果存在, componentWillReceivePropscomponentWillMountcomponentWillUpdate 将不会调用。 当你需要在以上老生命周期 setState 时, 我们更推荐你使用 getDerivedStateFromProps 方法,因为它能减少一次更新开销。

getSnapshotBeforeUpdate()

自 v1.3.0 起可用

  1. getSnapshotBeforeUpdate(prevProps, prevState)

<br />
getSnapshotBeforeUpdate()** 在最近一次渲染输出(提交到 DOM 节点)之前调用。**

它使得组件能在发生更改之前从 DOM 中捕获一些信息(例如,滚动位置)。

此生命周期的任何返回值将作为参数传递给 componentDidUpdate()

应返回 snapshot 的值(或 null)。

请注意: getSnapshotBeforeUpdate() 如果存在, componentWillReceivePropscomponentWillMountcomponentWillUpdate 将不会调用。 当你需要在以上老生命周期 setState 时, 我们更推荐你使用 getSnapshotBeforeUpdate 方法,因为它能减少一次更新开销。

componentWillMount()

  1. componentWillMount()

componentWillMount()** 在组件在装载发生前被立刻调用。

避免在该方法中引入任何的副作用或订阅。
对于这些使用场景,我们推荐使用 constructor() 来替代。

这是唯一的会在服务端渲染调起的生命周期钩子函数。

componentWillMount() 在小程序里对应的生命周期是 onLoad

componentDidMount()

  1. componentDidMount()

componentDidMount() 在组件被装载后立即调用。

初始化使得 DOM 节点应该进行到这里。
若你需要从远端加载数据,这是一个适合实现网络请求的地方
在该方法里 setState() 将会触发重新渲染。
**

componentWillReceiveProps()

  1. componentWillReceiveProps(nextProps)

componentWillReceiveProps() 在已经装载的组件接收到新属性前调用

若你需要更新状态响应属性改变,
你可能需对比 this.propsnextProps 并在该方法中使用 this.setState() 处理状态改变。

注意即使属性未有任何改变,Taro 可能也会调用该方法,
因此若你想要处理改变,请确保比较当前和之后的值。

在装载期间,Taro 并不会调用带有初始属性的 componentWillReceiveProps方法。
调用 this.setState 通常不会触发 componentWillReceiveProps

shouldComponentUpdate()

  1. shouldComponentUpdate(nextProps, nextState)

使用 shouldComponentUpdate() 让 Taro 知道组件的输出是否不受 state 或 props 当前变化的影响。

默认行为是在每次 state 更改时重新渲染,
并且在绝大多数情况下,你应该依赖于默认行为。

当接收到新的 props 或 state 时,shouldComponentUpdate() 在渲染之前被调用。

默认返回 true ,对于初始(第一次)渲染 或 使用 forceUpdate() 时,不调用此方法。

返回 false 不会阻止子组件在 state 更改时重新渲染。

componentWillUpdate()

  1. componentWillUpdate(nextProps, nextState)

当接收到新的 props 或 state 时,componentWillUpdate() 在渲染之前立即被调用。

在更新发生之前,使用这个方法可以作为执行准备更新的一个好机会。

这个方法在第一次渲染时不会被调用。

注意,这里不能调用 this.setState()
如果你需要更新 state 以响应 props 更改,请改用 componentWillReceiveProps()
如果你需要更新 state 来响应 props 的改变,可以使用 componentWillReceiveProps() 代替。

componentDidUpdate()

  1. componentDidUpdate(prevProps, prevState)

componentDidUpdate()更新发生后立即被调用
这个方法在第一次渲染时不会被调用。

componentWillUnmount()

  1. componentWillUnmount()

componentWillUnmount()一个组件被 卸载(unmounted) 和 销毁(destroyed) 之前立即被调用

在此方法中执行任何必要的清理,
例如使计时器无效,取消网络请求,清除一些可能会造成内存泄露的事件等。

在小程序中,一个挂在到 Page 组件的组件并不会执行 componentWillUnmount() 方法, 只有当他挂载的 Page 组件被销毁时,该组件才会执行 componentWillUnmount() 方法。

setState()

类属性

defaultProps

defaultProps 可以被定义为组件类的一个属性,用以为类设置默认的属性。
这对于未定义(undefined)的属性来说有用,而对于设为空(null)的属性并没用。
例如:

  1. class CustomButton extends React.Component {
  2. // ...
  3. }
  4. CustomButton.defaultProps = {
  5. color: 'blue'
  6. }

若未设置 props.color,其将被设置默认为 ‘blue’:

  1. render () {
  2. return <CustomButton /> // props.color will be set to blue
  3. }

若 props.color 设为 null,则其值则为 null:

  1. render() {
  2. return <CustomButton color={null} /> // props.color will remain null
  3. }

实例属性

props

this.props 包含了组件该调用者定义的属性。查看 组件 & Props 关于属性的介绍。

state

状态是该组件的特定数据,其可能改变多次。
状态由用户定义,且其应为纯 JavaScript 对象。

若你不在 render() 方法中使用它,那它就不应该被放在 state 中。

例如,你可直接将 timer IDs 放在实例上。

查看 生命周期 & State 了解更多关于状态的信息。

永远不要直接改变 this.state
因为调用 setState() 会替换你之前做的改变。将 this.state 当成不可变的。