Skip to main content

通过函数式分解和 Reducer 组合来重构 Reducer 逻辑

了解不同类型的子 reducer 函数是什么样子以及它们如何协同工作,可能会很有帮助。让我们来看一个演示,展示如何将一个大型的单一 reducer 函数重构为由几个较小函数组合而成。

注意:本示例故意采用冗长的样式以便于说明概念和重构过程,而不是追求代码的极致简洁。

初始 Reducer

假设我们的初始 reducer 是这样的:

const initialState = {
visibilityFilter: 'SHOW_ALL',
todos: []
}

function appReducer(state = initialState, action) {
switch (action.type) {
case 'SET_VISIBILITY_FILTER': {
return Object.assign({}, state, {
visibilityFilter: action.filter
})
}
case 'ADD_TODO': {
return Object.assign({}, state, {
todos: state.todos.concat({
id: action.id,
text: action.text,
completed: false
})
})
}
case 'TOGGLE_TODO': {
return Object.assign({}, state, {
todos: state.todos.map(todo => {
if (todo.id !== action.id) {
return todo
}

return Object.assign({}, todo, {
completed: !todo.completed
})
})
})
}
case 'EDIT_TODO': {
return Object.assign({}, state, {
todos: state.todos.map(todo => {
if (todo.id !== action.id) {
return todo
}

return Object.assign({}, todo, {
text: action.text
})
})
})
}
default:
return state
}
}

这个函数相当短,但已经逐渐变得复杂。我们在处理两个不同关注点(过滤器 vs 管理待办事项列表),嵌套使得更新逻辑难以阅读,而且整体上也不够清晰。

提取工具函数

一个好的第一步是提取一个工具函数,用来返回带有更新字段的新对象。还有一个重复的模式——尝试更新数组中某个特定项,也可以提取出一个函数:

function updateObject(oldObject, newValues) {
// 封装了传递一个新对象作为 Object.assign 第一个参数的思想
// 确保正确复制数据而不是进行变异
return Object.assign({}, oldObject, newValues)
}

function updateItemInArray(array, itemId, updateItemCallback) {
const updatedItems = array.map(item => {
if (item.id !== itemId) {
// 只想更新一个项目,所以保持其他项目不变
return item
}

// 使用提供的回调创建更新后的项目
const updatedItem = updateItemCallback(item)
return updatedItem
})

return updatedItems
}

function appReducer(state = initialState, action) {
switch (action.type) {
case 'SET_VISIBILITY_FILTER': {
return updateObject(state, { visibilityFilter: action.filter })
}
case 'ADD_TODO': {
const newTodos = state.todos.concat({
id: action.id,
text: action.text,
completed: false
})

return updateObject(state, { todos: newTodos })
}
case 'TOGGLE_TODO': {
const newTodos = updateItemInArray(state.todos, action.id, todo => {
return updateObject(todo, { completed: !todo.completed })
})

return updateObject(state, { todos: newTodos })
}
case 'EDIT_TODO': {
const newTodos = updateItemInArray(state.todos, action.id, todo => {
return updateObject(todo, { text: action.text })
})

return updateObject(state, { todos: newTodos })
}
default:
return state
}
}

这减少了重复代码,也使代码更易读。

提取 Case Reducers

接下来,可以把每个具体 case 拆成一个独立函数:

// 省略
function updateObject(oldObject, newValues) {}
function updateItemInArray(array, itemId, updateItemCallback) {}

function setVisibilityFilter(state, action) {
return updateObject(state, { visibilityFilter: action.filter })
}

function addTodo(state, action) {
const newTodos = state.todos.concat({
id: action.id,
text: action.text,
completed: false
})

return updateObject(state, { todos: newTodos })
}

function toggleTodo(state, action) {
const newTodos = updateItemInArray(state.todos, action.id, todo => {
return updateObject(todo, { completed: !todo.completed })
})

return updateObject(state, { todos: newTodos })
}

function editTodo(state, action) {
const newTodos = updateItemInArray(state.todos, action.id, todo => {
return updateObject(todo, { text: action.text })
})

return updateObject(state, { todos: newTodos })
}

function appReducer(state = initialState, action) {
switch (action.type) {
case 'SET_VISIBILITY_FILTER':
return setVisibilityFilter(state, action)
case 'ADD_TODO':
return addTodo(state, action)
case 'TOGGLE_TODO':
return toggleTodo(state, action)
case 'EDIT_TODO':
return editTodo(state, action)
default:
return state
}
}

现在,每个 case 的处理逻辑非常清晰。我们还可以开始看到一些模式。

按领域分离数据处理

我们的 appReducer 仍然处理了所有不同的应用情况。我们可以尝试把过滤逻辑和待办事项逻辑分开:

// 省略
function updateObject(oldObject, newValues) {}
function updateItemInArray(array, itemId, updateItemCallback) {}

function setVisibilityFilter(visibilityState, action) {
// 技术上,我们甚至不关心之前的状态
return action.filter
}

function visibilityReducer(visibilityState = 'SHOW_ALL', action) {
switch (action.type) {
case 'SET_VISIBILITY_FILTER':
return setVisibilityFilter(visibilityState, action)
default:
return visibilityState
}
}

function addTodo(todosState, action) {
const newTodos = todosState.concat({
id: action.id,
text: action.text,
completed: false
})

return newTodos
}

function toggleTodo(todosState, action) {
const newTodos = updateItemInArray(todosState, action.id, todo => {
return updateObject(todo, { completed: !todo.completed })
})

return newTodos
}

function editTodo(todosState, action) {
const newTodos = updateItemInArray(todosState, action.id, todo => {
return updateObject(todo, { text: action.text })
})

return newTodos
}

function todosReducer(todosState = [], action) {
switch (action.type) {
case 'ADD_TODO':
return addTodo(todosState, action)
case 'TOGGLE_TODO':
return toggleTodo(todosState, action)
case 'EDIT_TODO':
return editTodo(todosState, action)
default:
return todosState
}
}

function appReducer(state = initialState, action) {
return {
todos: todosReducer(state.todos, action),
visibilityFilter: visibilityReducer(state.visibilityFilter, action)
}
}

注意,因为这两个 “状态切片” 的 reducers 现在只接收它们自己对应的部分状态作为参数,它们不再需要返回复杂的嵌套状态对象,因此更简单了。

减少模板代码

差不多完成了。许多人不喜欢 switch 语句,所以常用一个函数创建一个从 action 类型映射到 case 函数的查找表。我们使用 减少模板代码 中介绍的 createReducer 函数:

// 省略
function updateObject(oldObject, newValues) {}
function updateItemInArray(array, itemId, updateItemCallback) {}

function createReducer(initialState, handlers) {
return function reducer(state = initialState, action) {
if (handlers.hasOwnProperty(action.type)) {
return handlers[action.type](state, action)
} else {
return state
}
}
}

// 省略
function setVisibilityFilter(visibilityState, action) {}

const visibilityReducer = createReducer('SHOW_ALL', {
SET_VISIBILITY_FILTER: setVisibilityFilter
})

// 省略
function addTodo(todosState, action) {}
function toggleTodo(todosState, action) {}
function editTodo(todosState, action) {}

const todosReducer = createReducer([], {
ADD_TODO: addTodo,
TOGGLE_TODO: toggleTodo,
EDIT_TODO: editTodo
})

function appReducer(state = initialState, action) {
return {
todos: todosReducer(state.todos, action),
visibilityFilter: visibilityReducer(state.visibilityFilter, action)
}
}

按状态切片合并 Reducers

最后一步,我们可以使用 Redux 内置的 combineReducers 工具,来处理顶层 appReducer 的状态切片逻辑。最终结果如下:

// 可复用的工具函数

function updateObject(oldObject, newValues) {
// 封装了传递一个新对象作为 Object.assign 第一个参数的思想
// 确保正确复制数据而不是进行变异
return Object.assign({}, oldObject, newValues)
}

function updateItemInArray(array, itemId, updateItemCallback) {
const updatedItems = array.map(item => {
if (item.id !== itemId) {
// 只想更新一个项目,所以保持其他项目不变
return item
}

// 使用提供的回调创建更新后的项目
const updatedItem = updateItemCallback(item)
return updatedItem
})

return updatedItems
}

function createReducer(initialState, handlers) {
return function reducer(state = initialState, action) {
if (handlers.hasOwnProperty(action.type)) {
return handlers[action.type](state, action)
} else {
return state
}
}
}

// 具体 case 的处理函数(“case reducer”)
function setVisibilityFilter(visibilityState, action) {
// 技术上,我们甚至不关心之前的状态
return action.filter
}

// 整个状态切片的处理函数(“slice reducer”)
const visibilityReducer = createReducer('SHOW_ALL', {
SET_VISIBILITY_FILTER: setVisibilityFilter
})

// case reducer
function addTodo(todosState, action) {
const newTodos = todosState.concat({
id: action.id,
text: action.text,
completed: false
})

return newTodos
}

// case reducer
function toggleTodo(todosState, action) {
const newTodos = updateItemInArray(todosState, action.id, todo => {
return updateObject(todo, { completed: !todo.completed })
})

return newTodos
}

// case reducer
function editTodo(todosState, action) {
const newTodos = updateItemInArray(todosState, action.id, todo => {
return updateObject(todo, { text: action.text })
})

return newTodos
}

// slice reducer
const todosReducer = createReducer([], {
ADD_TODO: addTodo,
TOGGLE_TODO: toggleTodo,
EDIT_TODO: editTodo
})

// “根 reducer”
const appReducer = combineReducers({
visibilityFilter: visibilityReducer,
todos: todosReducer
})

我们现在有了几个拆分 reducer 函数的示例:辅助工具函数如 updateObjectcreateReducer,具体 case 的处理函数如 setVisibilityFilteraddTodo,以及状态切片的处理函数如 visibilityReducertodosReducer。我们还看到 appReducer 是 “根 reducer” 的一个示例。

虽然最终结果相较于最初版本明显更长,但这主要是因为提取了工具函数、增加了注释以及为了清晰刻意写的详细代码,比如使用分开的 return 语句。从单个函数来看,它们的责任范围更小,意图也更明确。另外,在真实的项目中,这些函数大概率会被拆分到不同的文件中,例如 reducerUtilities.jsvisibilityReducer.jstodosReducer.jsrootReducer.js