我有一个待办事项列表react-redux应用程序,它试图让我了解redux的工作原理。我创建了一个初学者待办事项应用程序。

我遇到的问题是子组件ToDoList.jsx在从另一个文件中调用dispatch时不会更新,但是mapStateToProps确实会被调用,并返回正确的信息。该组件只是不重新渲染。

应用说明

有两个组件(app.js除外),AddToList.jsx(具有镜像文本字段的标签)和一个按钮,可用于将事物添加到待办事项列表数组中。

接下来,有一个名为ToDoList.jsx的子组件,它显示要执行的项目的数组。

CodeSandbox

App.js

...
  <div className="App">
    <Provider store={store}>
          <AddToList />
    </Provider>
  </div>
...


AddToList.jsx

...
    <>
        <input type="text" value={props.inputText} placeholder="To do item" onChange={props.inputChangeHandler}/>
        <button type="button" onClick={props.addToListHandler}>Add To List</button>
        <ToDoList />
    </>
...



function mapStateToProps(state) {
    return {
        inputText: state.inputText
    }
}

function mapDispatchToProps(dispatch) {
    return {
        inputChangeHandler: (evt) => {
            const action = {type: 'INPUT_CHANGE', text: evt.target.text};
            dispatch(action);
        },
        addToListHandler: () => {
            const action = {type: 'ADD_TO_LIST'};
            dispatch(action);
        }
    }
}

export default connect(
    mapStateToProps,
    mapDispatchToProps
)(AddToList);


ToDoList.jsx

import React from 'react';
import { connect } from 'react-redux';

const ToDoList = (props) => {

    return (
        <ul>
            {props.toDoList.map((toDoItem, index) => {
                return <li key={index}>{toDoItem}</li>
            })}
        </ul>
     );
}




function mapStateToProps(state) {
    console.log(state.toDoList);
    // ******* Gets Called, and logs correct data **********

    return {
        toDoList: state.toDoList
    };
}


export default connect(
    mapStateToProps
)(ToDoList);

最佳答案

问题是您正在使用以下代码:

case "ADD_TO_LIST":
  newState.toDoList.push(state.inputText);
  break;


由于您pushnewState.toDoList数组,所以reducer返回相同的对象(但被突变)。状态更改时,connect将为React组件提供相同的对象。但是,React不会更新列表,因为它是同一对象(通过引用)。

相反,您可以执行以下操作,而不会使原始对象发生变化:

case "ADD_TO_LIST":
  const toDoList = [...state.toDoList, state.inputText];
  newState = {
    ...state,
    toDoList
  };
  break;


请参阅此处的操作:https://codesandbox.io/s/11764qzpj(但是请注意,仍然存在一个错误!请尝试编辑输入并将其添加到列表中。我将作为练习留给您使用)。



旁注:我还注意到,每次调用reducer时都在创建一个newState。但这是不必要的,而且通常是有害的。如果状态没有变化,则应始终返回原始的state对象。否则,React会认为状态已更改,因此不必要地尝试重新渲染组件。

09-20 21:44
查看更多