美文网首页
setState详解

setState详解

作者: 张_何 | 来源:发表于2022-05-10 16:07 被阅读0次

setState

  • setState 是设置在 Component 原型上的方法,所有继承自Component的组件都可以调用该方法
    Component.prototype.setState = function(partialState, callback) {....}
  • 开发中我们并不能直接通过修改state的值来让界面发生更新, 因为我们修改了state之后,希望React根据最新的State来重新渲染界面,但是这种方式的修改React并不知道数据发生了变化,我们必须通过setState来告知React数据已经发生了变化

两种写法

对象式写法

setState(stateChange, [callBack]) ----- 对象式的 setState
1.stateChange为状态改变对象
2.callback 是可选的回调函数,它在状态更新完毕、界面也更新后(render 调用后)才被调用,

this.setState({count:count+1},()=>{
  console.log(this.setState.count);
})
函数式写法

setState(updater, [callBack])
1.updater 为返回stateChange 对象的函数
2.updater 可以接受 state 和 props
3.callback 是可选的回调函数,它在状态更新、界面也更新后(render调用后)才被调用

this.setState((state,props)=>{
  return {count:state.count+1}
},()=>{
  console.log(this.setState.count);
})
两种方式的使用原则:
  • 如果新状态不依赖于原状态,使用对象式
  • 如果新状态依赖于原状态,使用函数式
  • 如果需要在 setState() 执行后获取最新的状态数据,要在第二个 callback 函数中读取
  • 对象式的 setState 是函数式 setState 的简写方式(语法糖)
数据的合并
  • 通常 state 中保存的不止一个状态,而在我们使用setState 更新状态的时候,一般只更新一个或者部分几个,而不是全部更新,那么通过 setState 更新的最新状态会把state 中所有的状态给替换掉吗?
  • 不会,因为 setState 中源码里使用了 Object.assign({},prevState,partialState)将之前的状态和新的状态做了一个合并,这个函数将所有可枚举属性的值从一个或多个源对象复制到目标对象,如果有重复的属性,会使用后面的覆盖前面的属性值
状态是集合的更新
  • 当我们在 state 中保存的状态是集合类型,比如数组时, 在进行状态更新时,需要生成一个新的数组进行赋值,而不是直接在原数组上进行操作,比如增删改
 constructor(props) {
    super(props);
    this.state = {
      friends: [
        { name: "lilei", age: 20 },
        { name: "lily", age: 25 },
        { name: "lucy", age: 22 }
      ]
    }
  }
  insertData() {
    // 在开发中不要这样来做, 这样做,如果实现了shouldComponentUpdate,在方法里进行浅比较的话,认为两者是一样的,因为浅比较是比较两者的内存地址,
    // const newData = {name: "tom", age: 30}
    // this.state.friends.push(newData);
    // this.setState({
    //   friends: this.state.friends
    // });

    // 推荐做法
    const newFriends = [...this.state.friends];
    newFriends.push({ name: "tom", age: 30 });
    this.setState({
      friends: newFriends
    })
  }
  // 修改集合中某个对象的数据,也是要重新生成一个集合
  incrementAge(index) {
    const newFriends = [...this.state.friends];
    newFriends[index].age += 1;
    this.setState({
      friends: newFriends
    })
  }

setState 的更新

  • 有时候 setState 是异步更新的,有时候是同步更新的
setState的同步
  • 在 setTimeout 中,setState 是同步的
function changeText(){
  setTimeout(() => {
    this.setState({
      message:"你好啊,王小波"
    })
    console.log(this.state.message);// 你好啊,王小波
  }, 0);
}
  • 在原生 DOM 事件中,setState 是同步的
componentDidMount(){
  const btnEl = document.getElementById("btn");
  btnEl.addEventListener('click',()=>{
    this.setState({
      message:"你好啊,王小波"
    });
    console.log(this.state.message);// 你好啊,王小波
  })
}
setState的异步
  • 在组件的生命周期或 React 的合成事件中,setState 是异步的
changeText(){
    // message 的数据之前设置的是 "你好啊,李银河"
    this.setState({
      message:"你好啊,王小波"
    })
    console.log(this.state.message);// 你好啊,李银河
}
如果获取异步的结果
  • 在 setState 的第二个回调函数中获取
changeText(){
    // message 的数据之前设置的是 "你好啊,李银河"
    this.setState({
      message:"你好啊,王小波"
    }, ()=>{
      console.log(this.state.message);// 你好啊,王小波
    })
    console.log(this.state.message);// 你好啊,李银河
}
  • componentDidUpdate函数中获取
setState 同步或异步的源码分析
  • 通过查看源码我们看到 setState 是设置在 Component 上的方法
  • 然后调用 this.updater.enqueueSetState(this, partialState, callback, 'setState') 这里的updater是在构造器中传过来的
  • 上面传入的 update 其实是 react-reconciler 中 ReactFiberClassComponent 中的 classComponentUpdater
const classComponentUpdater = {
  isMounted,
  enqueueSetState(inst, payload, callback) {
    const fiber = getInstance(inst);
    const currentTime = requestCurrentTimeForUpdate();
    const suspenseConfig = requestCurrentSuspenseConfig();
    const expirationTime = computeExpirationForFiber(
      currentTime,
      fiber,
      suspenseConfig,
    );

    const update = createUpdate(expirationTime, suspenseConfig);
    update.payload = payload;
    if (callback !== undefined && callback !== null) {
      if (__DEV__) {
        warnOnInvalidCallback(callback, 'setState');
      }
      update.callback = callback;
    }
    enqueueUpdate(fiber, update);
    scheduleWork(fiber, expirationTime);
  },
  ....
};
  • 在获取expirationTime的函数computeExpirationForFiber中可以看到设置的执行的优先级, 如果优先级设置的是 sync,那么就是同步执行的
export function computeExpirationForFiber(
  currentTime: ExpirationTime,
  fiber: Fiber,
  suspenseConfig: null | SuspenseConfig,
): ExpirationTime {
  const mode = fiber.mode;
  if ((mode & BlockingMode) === NoMode) {
    return Sync;
  }

  const priorityLevel = getCurrentPriorityLevel();
  if ((mode & ConcurrentMode) === NoMode) {
    return priorityLevel === ImmediatePriority ? Sync : Batched;
  }

  if ((executionContext & RenderContext) !== NoContext) {
    // Use whatever time we're already rendering
    // TODO: Should there be a way to opt out, like with `runWithPriority`?
    return renderExpirationTime;
  }

  let expirationTime;
  if (suspenseConfig !== null) {
    // Compute an expiration time based on the Suspense timeout.
    expirationTime = computeSuspenseExpiration(
      currentTime,
      suspenseConfig.timeoutMs | 0 || LOW_PRIORITY_EXPIRATION,
    );
  } else {
    // Compute an expiration time based on the Scheduler priority.
    switch (priorityLevel) {
      case ImmediatePriority:
        expirationTime = Sync;
        break;
      case UserBlockingPriority:
        // TODO: Rename this to computeUserBlockingExpiration
        expirationTime = computeInteractiveExpiration(currentTime);
        break;
      case NormalPriority:
      case LowPriority: // TODO: Handle LowPriority
        // TODO: Rename this to... something better.
        expirationTime = computeAsyncExpiration(currentTime);
        break;
      case IdlePriority:
        expirationTime = Idle;
        break;
      default:
        invariant(false, 'Expected a valid priority level');
    }
  }

相关文章

  • 关于React中的setState

    前言 在看React setState 详解中看到一道有意思的题目,关于setState函数中state的值的问题...

  • setState详解

    首先如果直接在setState后面获取state的值是获取不到的如图: 第一次,第二次打印都是初始值。通过异步方法...

  • setState详解

    setState setState 是设置在 Component 原型上的方法,所有继承自Component的组件...

  • Component.setState()详解

    setState() API setState()方法主要是用来更新组件状态,让组件重新渲染。 应该将setSta...

  • react的setState使用详解

    react修改状态是不能直接修改的,需要使用setState来进行状态的修改,但是setState的使用会存在一些...

  • react_08setState详解

    (一)为什么使用setState? 发现会有警告,并且页面的数字并不会更新。 : 为啥能使用setState方法?...

  • react拓展

    setState() setState更新状态的2种方式 对象式的setState 函数式的setState 对象...

  • setState()状态更新函数

    理解setState的关键 setState不会立刻改变React组件中state的值; setState通过引发...

  • 08react基础-react原理

    setState()更新数据 setState()更新数据是异步的 注意:使用该语法,后面的setState不要依...

  • 『react』setState()特性

    一.setState()更新数据 由于setState()更新数据是异步的,所以后面的setState()不要依赖...

网友评论

      本文标题:setState详解

      本文链接:https://www.haomeiwen.com/subject/rhoturtx.html