在redux中修改响应数据的合理位置在哪里?

时间:2016-03-31 07:37:52

标签: reactjs redux react-redux redux-thunk

使用React,Redux和Thunk的组合,我有以下内容:

actions.js

import $ from 'jquery';
import * as types from '../constants/ActionTypes';
import { API_PATH } from '../constants/Config';

export function coursesLoaded(courses) {
    return { type: types.COURSES_LOADED, courses };
}

export function fetchData() {
    return (dispatch) => {
        return $.getJSON(API_PATH).then((response) => {
            dispatch(coursesLoaded(response.result));
        });
    };
}

reducer.js

import { routerReducer as routing } from 'react-router-redux';
import { combineReducers } from 'redux';
import * as types from '../constants/ActionTypes';

const initialState = {
    courses: [],
};

function main(state = initialState, action) {
    switch(action.type) {
        case types.COURSES_LOADED:
            return {
                ...state,
                courses: action.courses,
            };
        default:
            return state;
    }
}

const rootReducer = combineReducers({ main, routing });

export default rootReducer;

上面的两个片段坐得很好,我觉得它们符合Redux的意图。我现在想要在响应容器之前对响应中返回的字段进行一些修改。

例如,响应可能是:

[
    { code: "R101", name: "Intro to Redux", author: "Dan" },
    { code: "R102", name: "Middleware", author: "Dan" },
]

我想将其更改为(简单的简单示例):

[
    { code: "R101", name: "Intro to Redux", author: "Dan", additionalProperty: "r101_intro_to_redux" },
    { code: "R102", name: "Middleware", author: "Dan", additionalProperty: "r102_middleware" },
]

到目前为止的研究

选项一 看看Redux上的异步示例,我可以看到这里有一个轻松的响应: https://github.com/reactjs/redux/blob/master/examples/async/actions/index.js#L33

选项二 看看其他Stackoverflow问题,它让我相信保持它的行为更有意义,因为reducers应该是什么修改状态(但也许这并不算作状态?): Redux - where to prepare data

选项三 我有一个倾向,这是中间件的工作 - 这是normalizr如何处理它,但我找不到任何非被动中间件的例子。如果中间件就在这里,中间件应该调度某种SET_STATE动作,还是可以在中间件中自由更新状态?

修改

使用一些中间件进行实验,例如:

import { lowerCase, snakeCase } from 'lodash';
import * as types from '../constants/ActionTypes';

    export default store => next => action => {
        if(action.type == types.COURSES_LOADED) {
            action.courses = action.courses.map((course) => {
                course.additionalProperty = snakeCase(lowerCase(`${course.code} ${course.name}`));
                return course;
            });
        }
        return next(action);
    }

似乎工作正常 - 这确实是中间件的意图吗?原始问题成立 - 它是理想的地方吗?

2 个答案:

答案 0 :(得分:11)

至于我,我在行动中做了这类事情(coursesLoadedfetchData)。

以下是原因:

  • 这不是商店材料,这只是外部数据管理,所以与应该更改商店STATE的Reducer无关
  • 不同的Reducer实际上可能需要相同的校正数据,例如,您可以使用另一个缩减器来收集所有additionalProperty用于实现目的,因此在操作中执行此操作可确保将正确的数据发送到所有Reducer。 / LI>
  • 这不是中间件的典型工作,这仅适用于一个操作,而中间件如果通过一堆操作以相同的方式使用则会很有用。此外,使用中间件更加模糊,并将其与读者分开。有行动 - >减速机更简单,没有任何重大缺点。

答案 1 :(得分:0)

或者您可以使用selector,并将原件保存在redux存储中。

const mapStateToProps = state => {
  courses: mySelectorThatAddsFields(state.courses),
}