redux/react 应用程序中的状态有一个名为 reducer 的属性

IT技术 javascript reactjs state redux
2021-05-04 09:48:32

我正在使用 Redux 和 React 创建一个应用程序。我遇到了一个问题,我无法将状态映射到组件属性,因为状态具有与我使用的减速器名称相匹配的属性。

根减速器是用combineReducers方法创建的

const rootReducer = combineReducers({
  appReducer
});

初始状态是

const initialState = {
  sources: [], 
  left: {}, 
  right: {},
  diff: {} 
}

但是在组件函数中mapStateToProps

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

state.sourcesundefined因为state参数的值

{
  appReducer: {
    sources: [], 
    left: {}, 
    right: {}, 
    diff: {}
  }
}

这是 redux 的特性吗?那么当我使用更多的减速器时,它们都会为state变量添加新属性吗?或者我这边有什么问题(我从未在 redux 教程中注意到这种行为)。

谢谢

2个回答

如果您只有一个减速器,则不需要combineReducers(). 直接使用就可以了:

const initialState = {
  sources: [],
  left: {},
  right: {}
}
function app(state = initialState, action) {
  switch (action.type) {
  case 'ADD_SOURCE':
    return Object.assign({}, state, {
      sources: [...state.sources, action.newSource]
    })
  case 'ADD_SOURCE_TO_LEFT':
    return Object.assign({}, state, {
      left: Object.assign({}, state.left, {
        [action.sourceId]: true
      })
    })
  case 'ADD_SOURCE_TO_RIGHT':
    return Object.assign({}, state, {
      right: Object.assign({}, state.right, {
        [action.sourceId]: true
      })
    })
  default:
    return state
  }
}

现在您可以使用该减速器创建一个商店:

import { createStore } from 'redux'
const store = createStore(app)

并将一个组件连接到它:

const mapStateToProps = (state) => ({
  sources: state.sources
})

然而,你的减速器很难阅读,因为它一次更新了许多不同的东西。现在,是您想要将其拆分为几个独立的减速器的时刻:

function sources(state = [], action) {
  switch (action.type) {
  case 'ADD_SOURCE':
    return [...state.sources, action.newSource]
  default:
    return state
  }
}

function left(state = {}, action) {
  switch (action.type) {
  case 'ADD_SOURCE_TO_LEFT':
    return Object.assign({}, state, {
      [action.sourceId]: true
    })
  default:
    return state
  }    
}

function right(state = {}, action) {
  switch (action.type) {
  case 'ADD_SOURCE_TO_RIGHT':
    return Object.assign({}, state, {
      [action.sourceId]: true
    })
  default:
    return state
  }    
}

function app(state = {}, action) {
  return {
    sources: sources(state.sources, action),
    left: left(state.left, action),
    right: right(state.right, action),
  }
}

这更容易维护和理解,也更容易独立更改和测试 reducer。

最后,作为最后一步,我们可以使用combineReducers()生成 root appreducer 而不是手工编写:

// function app(state = {}, action) {
//   return {
//     sources: sources(state.sources, action),
//     left: left(state.left, action),
//     right: right(state.right, action),
//   }
// }

import { combineReducers } from 'redux'
const app = combineReducers({
  sources,
  left,
  right
})

使用combineReducers()而不是手动编写 root reducer没有什么大的好处,只是它的效率稍微高一些,并且可能会为您节省一些拼写错误。此外,您可以在您的应用程序中多次应用此模式:以嵌套方式将不相关的减速器多次组合到单个减速器中是可以的。

所有这些重构都不会对组件产生影响。

我建议您观看我关于 Redux 的免费 Egghead 课程,课程涵盖了这种reducer 组合模式并展示了如何combineReducers()实现。

实际上,我相信您的初始状态是:

{
  appReducer: {
    sources: [],
    left: {},
    right: {},
    diff: {}
  }
}

这是因为combineReducers通过获取 reducer 的名称并将其内容映射到该名称来工作。

另外,只是一个说明,但是如果您要使用 1 个以上的减速器,则减速器的名称应该比 更具体appReducer,并且(仅我个人的意见)他们不需要这个词reducer典型的应用程序可能如下所示:

combineReducers({
  user: userReducer,
  messages: messagesReducer,
  notifications: notificationsReducer
});

然后,您的状态可以像这样访问:

state.user.email
state.messages[0]