reactjs - 更新嵌套对象时 UseState 不重新渲染

标签 reactjs react-hooks

我通过将数据推送到旧状态对象并将其作为值返回来更新 useEffect。

此代码实际上是从 useState() 更改 _series 变量,但没有发生重新渲染,为什么?

import { TimeSeries, Pipeline, Stream, EventOut, TimeEvent, TimeRange  } from "pondjs";

export default () => {

  const [_series, $series] = useState(()=>{
    let state = { data : { "name": "x", "columns": ["time", "value"], "points": [], "i" : 0}}
    for(let i=10; i >= 0; i--){state.data.points.push( [new Date(i)-(i*100), Math.round(Math.random()*100)])}
    return state;
  })

  useEffect(() => {
    const interval = setInterval(() => {
      $series(s => {
        s.data.points.push( [new Date(s.data.i*1000), Math.round(Math.sin(s.data.i/10)*50+50)] )
        s.data.points.shift();
        s.data.i++;
        return s;
      });
    }, 500);
  }, []);


    return(
    <p>{

          JSON.stringify((new TimeSeries(_series.data)).collection())

    }</p>
    )

}


我也可能在更新状态中的键“i”时犯了一个错误,所以我尝试一次更新对象的多个部分。这是不好的做法吗?

最佳答案

问题是,当您通过修改原始状态来更改状态时,它的值会在原始引用处更新,因此 react 认为没有任何改变,因此它不会重新渲染,这就是为什么期望对待当您尝试更新状态时,状态就好像它是不可变的

 const { useState, useEffect } = React;
 const App = () => {
    
      const [_series, $series] = useState(()=>{
        let state = { data : { "name": "x", "columns": ["time", "value"], "points": [], "i" : 0}}
        for(let i=10; i >= 0; i--){state.data.points.push( [new Date(i)-(i*100), Math.round(Math.random()*100)])}
        return state;
      })
    
      useEffect(() => {
        const interval = setInterval(() => {
          $series(s => {
            return {
                ...s,
                data: {
                  ...s.data,
                  i: s.data.i + 1,
                  points: [...s.data.points.slice(1), [new Date(s.data.i*1000), Math.round(Math.sin(s.data.i/10)*50+50)]]
                }
            }
          });
        }, 500);
      }, []);
    
    
        return(
        <p>{
    
              JSON.stringify(_series.data)
    
        }</p>
        )
    
    }
    
ReactDOM.render(<App/>, document.getElementById('app'));
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.8.3/umd/react.production.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.8.3/umd/react-dom.production.min.js"></script>
<div id="app" />

关于reactjs - 更新嵌套对象时 UseState 不重新渲染,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56033178/

相关文章:

javascript - 我可以从 setTimeout() 调用 setState() 吗?

ReactJs ErrorBoundary 在最简单的情况下不起作用

javascript - 编写一些从 jQuery 到 JS (React JS) 的代码

javascript - 在React组件文件夹中编译SASS文件

javascript - 在 React Native 中根据条件渲染元素

javascript - React 检测到 Hooks 的调用顺序发生了变化,但我看不到任何有条件调用的 hooks

javascript - React-native 异步获取返回 null

reactjs - 如何解决 React "too many re-renders"错误

javascript - 在 Prop 中传递对象时避免重新渲染 child

javascript - react useEffect Hook 与长时间运行的任务和合并状态