将 Redux 添加到现有的 React 应用程序

IT技术 reactjs redux react-redux
2021-04-24 07:49:52

我一直在研究 React 应用程序,并且已经到了需要 Redux 来处理它的某些方面的地步。

在阅读了一堆教程后,我一直在思考如何使我的“更智能”组件“变笨”并将函数移动到我的动作和减速器中。

因此,例如,该应用程序的一个方面更像是待办事项列表样式。

我的一堂课是这样开始的:

export default class ItemList extends React.Component {
  constructor() {
    super();
    this.state = { items: [],
                   completed: [],
                  };
    this.addItem = this.addItem.bind(this);
    this.completeItem = this.completeItem.bind(this);
    this.deleteItem = this.deleteItem.bind(this);
  }

  addItem(e) {
    var i = this.state.items;
    i.push({
      text: this._inputElement.value,
      paused: false,
      key: Date.now()
    });
    this.setState({ items: i });
    e.preventDefault();
    this._inputElement.value = '';
    this._inputElement.focus();
  }

  completeItem(e) {
    this.deleteItem(e);
    var c = this.state.completed;
    c.push({
      text: e.target.parentNode.parentNode.getElementsByClassName('item-name')[0].innerHTML,
      paused: false,
      key: Date.now()
    });
    this.setState({ completed: c });
  }

  deleteItem(e) {
    var i = this.state.items;
    var result = i.filter(function(obj) {
        return obj.text !== e.target.parentNode.parentNode.getElementsByClassName('item-name')[0].innerHTML;
    });
    this.setState({ items: result });
  }

  // ... more irrelevant code here ...

  // there's a function called createTasks that renders individual items

  render() {
    var listItems = this.state.items.map(this.createTasks);

    return <div className="item-list">
      <form className="form" onSubmit={this.addItem}>
        <input ref={(a) => this._inputElement = a}
               placeholder="Add new item"
               autoFocus />
        <button type="submit"></button>
      </form>

      {listItems}
    </div>;
  }
}

所以,正如你所看到的,它的逻辑非常重。我已经开始通过<Provider>在我的索引文件中添加 a添加 Redux ,并制作了一个到目前为止相当空的基本 reducers 文件:

import { combineReducers } from 'redux';

const itemList = (state = {}, action) => {

};

// ... other irrelevant reducers

const rootReducer = combineReducers({
  itemList,
  // ...
});

export default rootReducer;

...而且我已经制作了一个动作文件,其中也没有太多内容。

我一直在努力弄清楚:

  • 我见过的大多数操作示例都只返回某种 JSON,我在使用组件可以使用的 JSON 的 reducer 中返回什么?
  • 我的组件逻辑有多少是可重用的,还是我应该忘记它?重用尽可能多的代码的最佳方法是什么?
1个回答

首先,您需要了解 redux 如何与 react 协同工作的整体图景。

在开始之前,让我们先了解什么是智能组件和哑组件。

智能组件

  1. 你所有的代码逻辑都需要在这里处理
  2. 它们也称为容器。
  3. 它们与商店(又名状态管理)交互以更新您的组件。

哑组件

  1. 他们只是从你的容器中读取props并渲染你的组件
  2. 这只是 UI 视图,不应包含任何逻辑。
  3. 所有样式/html/css 都在您的愚蠢组件中。

是一篇很棒的文章,如果您仍有疑问,可以阅读它以了解智能和愚蠢的组件。

好的,现在让我们尝试了解 redux 的工作原理:-

  • 您的智能组件(又名容器)与您的 redux 存储交互
  • 您从容器中触发动作。
  • 你的行为调用你的 api
  • 您的操作结果通过减速器更新商店
  • 您的容器通过 mapStateToProps 函数读取存储,一旦存储中的值发生变化,它就会更新您的组件。

现在让我们考虑您的待办事项示例

TodoListContainer.js

class TodoListContainer extends Component {

  componentWillMount () {
    // fire you action action
  }


  render () {
    return (
      <Todos todos=={this.props.todos} />
    )
  }
}


function mapStateToProps(state) {
  const {todos} = state;
  return {
    todos;
  }
}

export default connect(mapStateToProps)(TodoListContainer)

TodoList.js

class TodoList extends Component {

  renderTodos() {
    return this.props.todos.map((todo)=>{
      return <Todo todo={todo} key={todo.id} />
    })
  }

  render () {
    return () {
      if (this.props.todos.length === 0) {
        return <div>No todos</div>
      }
      return (
        <div>
          {this.renderTodos()}
        </div>
      )
    }
  }
}

export default class TodoList

Todo.js

class Todo extends Component {

  render () {
    return (
      <div>
        <span>{this.props.todo.id}</span>
        <span>{this.props.todo.name}</span>
      </div>
    )
  }
}

减速器

export default function todos(state={},action) {
  switch (action.type) {
    case 'RECEIVE_TODOS':
       return Object.assign(state,action.todos);
  }
}

行动

function fetchTodos() {
  return(dispatch) => {
      axios.get({
    //api details
    })
    .then((res)=>{
      dispatch(receiveTodos(res.todos))
    })
    .catch((err)=>{
      console.warn(err)
    })
  }
}

function receiveTodos(todos) {
  return {
    type: 'RECEIVE_TODOS',
    todos
  }
}

现在,如果您已阅读 redux 文档,您会看到操作返回对象,那么我将如何在那里调用我的 api,它返回一个函数而不是一个对象。为此,我使用了 redux thunk,你可以在这里阅读

我给了你一个例子,你可以在其中获取待办事项。如果您想执行其他操作,例如 deleteTodo、addTodo、modifyTodo,那么您可以在适当的组件中执行此操作。

  1. DeleteTodo - 您可以在 TodoListContainer 中进行。
  2. 添加Todo - 您可以在TodoListContainer 中进行。
  3. 更改状态(已完成/待处理) - 您可以在 TodoListContainer 中进行。
  4. ModifyingTodo - 您可以在 TodoContainer 中进行。

您也可以在此处查看详细示例,但在此之前我会说应该先了解 redux 的基础知识,您可以在此处找到

PS:我即时编写了代码,因此它可能无法正常工作,但只需稍加修改即可工作。