我们使用两种数据来控制一个组件:props和state。props是在父组件中指定,而且一经指定,在被指定的组件的生命周期中则不再改变。 对于需要改变的数据,我们需要使用state。
state属性主要用来存储组件自身需要的数据,是组件自己私有的,我们一般通过修改 state 属性的值来更新数据,React 内部会监听 state
的变化,一旦发生变化就会主动触发组件的 render() 方法来更新 Dom 结构
state的使用
一般来说,你需要在 constructor()方法中初始化 state(这是ES6的写法,ES5 中一般在 getInitialState()
方法中来初始化 state
),然后在需要修改时调用 setState() 方法。
不要使用 this.state
来修改 state
属性值,应该调用 setState()
方法,this.state
是无效的。
setState()方法
setState()
的完整表达式:
setState(updater, [callback])
setState()
方法会把对组件 state 的改变加入到队列中,并且告诉 React 这个组件及其子组件需要重新渲染。
第一个参数:updater
函数
setState(updater, callback)
方法的第一个参数是一个固定格式的 updater
函数:
1 | (prevState, props) => stateChange |
prevState
是一个对之前状态(previous state)的引用,我们是不能直接修改这个参数的值,要想修改 state
的值,我们应该根据 prevState
和 props
参数来创建一个新的 JavaScript 对象。
例子1:
1 | this.setState((prevState, props) => { |
你也可以传一个对象而不是函数,来作为setState(updater, callback)
方法的第一个参数,React 会将该参数 merge 到 state 中。
例子2:
1 | this.setState({quantity: 2}); |
第二个参数:callback
setState(updater, callback)
方法的第二个参数 callback
是一个可选参数。(基本是空的,一般不这么使用)
为了更好的性能表现,React 并不能保证 setState()
一被调用 state 就能更新。所以,如果在调用 setState()
之后,马上就读取 this.state
的值的话,可能会出现误差。
因此,这种情况下,推荐使用 componentDidUpdate
或者 setState(updater, callback)
方法的 callback
来获取最新的状态。
React 官方更推荐使用 componentDidUpdate()
,而不是 callback
来监听 update 事件(注: 除非 shouldComponentUpdate()
方法返回 false
,setState()
将永远都会引发重新渲染)。
例子1:
1 | import React, {Component} from 'react'; |
如上图效果:
- shouldComponentUpdate 不重写或者返回true,则象左侧ios模拟器一样,可以修改状态
- 返回false,则不重写
- 官方不推荐我们使用setState的第二个参数Callback的形式获取,我们可以在componentDidUpdate回调中,获取最新的State
如何正确操作state
不要使用this.state
来修改state
1 | // Wrong |
记住,this.state
是不可变的。
1 | //事实证明,并改变不了 |
应该调用 setState()
方法:
1 | // Correct |
state的改变是一个覆盖(合并)过程
当你调用 setState()
方法时,React 会将将你当前所提供的对象合并到当前的状态中。
例子1:
1 | constructor(props) { |
⬆面的栗子,state更新至替换了comments,posts不受影响
state传递?
注意:
- 父组件和子组件之间不能通过 state 来交互,父组件只能将自己的 state 值传给子组件的 props。
- 这种数据传递的方式通常被称为 “自顶向下(top-down)”或者“单向(unidirectional)”数据流。
- 任何state都是由一个特定的组件所拥有的,任何state数据修改默认只会影响当前组件,
- 可以通过回调传值(后面会整理)
- React Native建议由顶层的父组件定义state值,并将state值作为子组件的props属性值传递给子组件,这样可以保持单一的数据传递。