高级教程: Redux工具包 实践
在中级教程中,你看到了如何在一个典型的基本 React 应用中使用 Redux工具包,同时还有如何把已有的纯 Redux 代码进行转换以使用 RTK。 另外,你还看到了如何在 reducer 函数中编写“可变的”immutable 更新代码,以及如何编写一个为了生成 action payload 的 “prepare 回调函数“。
在本教程中,你将会看到如何在一个比 todo 列表更大的“实际”应用中使用 Redux工具包。本教程会展示几个概念:
- 如何将一个”纯 React“应用转化以使用 Redux
- 异步逻辑,例如获取数据,是如何融入到 RTK 中的
- 如何结合 TypeScript 来使用 RTK
在此过程中,我们探究几个使用了 Typescript 技术编写的例子的,以提高你的代码质量,同时我们还会看到如何使用新的 React-Redux hooks APIs , 作为传统的 connect
API的另一个替代品。
注意: 本教程并不是关于 TypeScript 通用或者在 Redux 中特定使用方式的完整教程,且这里演示的示例并没有达到 100% 的类型安全。想要获取更多信息,请查阅社区的资源,例如 React TypeScript Cheatsheet 和 React/Redux TypeScript Guide。
此外,本教程并不意味着你 必须 把 React 应用的逻辑完全地转换成 Redux 的逻辑。这取决于你如何选择哪些状态应当留在 React 组件中,哪些应该放到 Redux 里。教程的示例仅仅向你展示 能以什么方式 把逻辑换成 Redux 逻辑,如果你选择这么做的话。
本教程中,实现整个应用的完整源代码可以从 github.com/reduxjs/rtk-github-issues-example 获得。我们将逐步解释整个转换的过程,正如仓库里的历史记录所展示一样。有其作用的独立提交的链接,将像如下高亮的引用块显示:
- 这里是提交信息
回顾示例应用开始过程
本教程的示例应用是一个 Github Issues 查找应用。它可以让用输入 Github 里面的某一个组织及其仓库的名字、获取现有的 open issues 列表、分页浏览 issues 列表,以及查看每一条具体 issue 的内容和评论。
这个应用的的第一次提交是一个纯 React 的实现,其中用到了带有负责管理状态和副作用(例如获取数据等)的 hooks 的函数式组件。代码已经使用了 TypeScript 编写,并且样式是通过 CSS Modules 完成的。
Let's start by viewing the original plain React app in action:
React 代码库源码概览
代码仓库已经使用了“功能文件夹”结构编写,主要组成部分有:
/api
: fetching functions and TS types for the Github Issues API/app
: main<App>
component/components
: components that are reused in multiple places/features
/issueDetails:
components for the Issue Details page/issuesList
: components for the Issues List display/repoSearch
: components for the Repo Search form
/utils
: various string utility functions
Setting Up the Redux Store
Since this app doesn't yet use Redux at all, the first step is to install Redux Toolkit and React-Redux. Since this is a TypeScript app, we'll also need to add @types/react-redux
as well. Add those packages to the project via either Yarn or NPM.
Next, we need to set up the usual pieces: a root reducer function, the Redux store, and the <Provider>
to make that store available to our component tree.
In the process, we're going to set up "Hot Module Replacement" for our app. That way, whenever we make a change to the reducer logic or the component tree, Create-React-App will rebuild the app and swap the changed code into our running app, without having to completely refresh the page.
Creating the Root Reducer
First, we'll create the root reducer function. We don't have any slices yet, so it will just return an empty object.
However, we're going to want to know what the TypeScript type is for that root state object, because we need to declare what the type of the state
variable is whenever our code needs to access the Redux store state (such as in mapState
functions, useSelector
selectors, and getState
in thunks).
We could manually write a TS type with the correct types for each state slice, but we'd have to keep updating that type every time we make any change to the state structure in our slices. Fortunately, TS is usually pretty good at inferring types from the code we've already written. In this case, we can define a type that says "this type is whatever gets returned from rootReducer
", and TS will automatically figure out whatever that contains as the code is changed. If we export that type, other parts of the app can use it, and we know that it's up to date. All we have to do is use the built-in TS ReturnType
utility type, and feed in "the type of the rootReducer
function" as its generic argument.
app/rootReducer.ts
Note: For other ways to infer the
RootState
, view the Usage with TypeScript guide
Store Setup and HMR
Next, we'll create the store instance, including hot-reloading the root reducer. By using the module.hot
API for reloading, we can re-import the new version of the root reducer function whenever it's been recompiled, and tell the store to use the new version instead.
app/store.ts
The require('./rootReducer').default
looks a bit odd. That's because we're mixing CommonJS synchronous import syntax with ES modules, so the "default export" is in a object field called default
. We could probably also have used import()
and handled the reducer replacement asynchronously as well.
Provider
Rendering the Now that the store has been created, we can add it to the React component tree.
As with the root reducer, we can hot-reload the React component tree whenever a component file changes. The best way is to write a function that imports the <App>
component and renders it, call that once on startup to show the React component tree as usual, and then reuse that function any time a component is changed.
index.tsx
Converting the Main App Display
With the main store setup done, we can now start converting the actual app logic to use Redux.
Evaluating the Existing App State
Currently, the top-level <App>
component uses React useState
hooks to store several pieces of info:
- The selected Github org and repo
- The current issues list page number
- Whether we're viewing the issues list, or the details for a specific issue
Meanwhile, the <RepoSearchForm>
component also uses state hooks to store the work-in-progress values for the controlled form inputs.
The Redux FAQ has some rules of thumb on when it makes sense to put data into Redux. In this case, it's reasonable to extract the state values from <App>
and put those into the Redux store. While there's only one component that uses them now, a larger app might have multiple components that care about those values. Since we've set up HMR, it would also be helpful to persist those values if we make future edits to the component tree.
On the other hand, while we could put the WIP form values into the Redux store, there's no real benefit to doing so. Only the <RepoSearchForm>
component cares about those values, and none of the other rules of thumb apply here. In general, most form state probably shouldn't be kept in Redux. So, we'll leave that alone.
Creating the Initial State Slices
The first step is to look at the data that is currently being kept in <App>
, and turn that into the types and initial state values for our "issues display" slice. From there, we can define reducers to update them appropriately.
Let's look at the source for the whole slice, and then break down what it's doing:
features/issuesDisplay/issuesDisplaySlice.ts
State Contents Type Declarations
The org and repo values are simple strings, and the current issues page is just a number. We will use a union of string constants to indicate if we're showing the issues list or the details of a single issue, and if it's the details, we need to know the issue ID number.
We can define types for a couple of those pieces by themselves for reuse in the action types later, and also combine them into a larger type for the entire state we plan to track.
The "current display" part requires a bit of extra work, because the type listed for the state includes a page number, but the UI won't include one when it dispatches an action to switch to the issues list. So, we define a separate type for that action's contents.
Declaring Types for Slice State and Actions
createSlice
tries to infer types from two sources:
- The state type is based on the type of the
initialState
field - Each reducer needs to declare the type of the action it expects to handle
The state type is used as the type for the state
parameter in each of the case reducers and the return type for the generated reducer function, and the action types are used for the corresponding generated action creators. (Alternately, if you define a "prepare callback" alongside a reducer, the prepare callback's arguments are used for the action creator too, and the return value from the callback must match the declared type for the action the reducer expects.)
The main type you will use when declaring action types in reducers is PayloadAction<PayloadType>
. createAction
uses this type as its return value.
Let's look at a specific reducer as an example:
We don't have to declare a type for state
, because createSlice
already knows that this should be the same type as our initialState
: the CurrentDisplayState
type.
We declare that the action object is a PayloadAction
, where action.payload
is a number
. Then, when we assign state.page = action.payload
, TS knows that we're assigning a number to a number, and it works correctly. If we were to try calling issuesDisplaySlice.actions.setCurrentPage()
, we would need to pass a number in as the argument, because that number will become the payload in the action.
Similarly, for displayRepo(state, action: PayloadAction<CurrentRepo>)
, TS knows that action.payload
is an object with org
and repo
string fields, and we can assign them to the state. (Remember that these "mutative" assignments are only safe and possible because createSlice
uses Immer inside!)
Using the Slice Reducer
As with other examples, we then need to import and add the issues display slice reducer to our root reducer:
app/rootReducer.ts
Converting the Issues Display
Now that the issues display slice is hooked up to the store, we can update <App>
to use that instead of its internal component state.
We need to make three groups of changes to the App
component:
- The
useState
declarations need to be removed - The corresponding state values need to be read from the Redux store
- Redux actions need to be dispatched as the user interacts with the component
Traditionally, the last two aspects would be handled via the React-Redux connect
API. We'd write a mapState
function to retrieve the data and a mapDispatch
function to hold the action creators, pass those to connect
, get everything as props, and then call this.props.setCurrentPage()
to dispatch that action type.
However, React-Redux now has a hooks API, which allows us to interact with the store more directly. useSelector
lets us read data from the store and subscribe to updates, and useDispatch
gives us a reference to the store's dispatch
method. We'll use those throughout the rest of this tutorial.
First, we'll import the necessary functions, plus the RootState
type we declared earlier, and remove the hardcoded default org and repo strings.
app/App.tsx
Next, at the top of App
, we'll remove the old useState
hooks, and replace them with a call to useDispatch
and useSelector
:
We pass a "selector" function into useSelector
, which is just a function that accepts our Redux store state as its parameter and returns some result. We declare that the type of the state
argument is the RootState
type we defined over in the root reducer, so that TS knows what fields are inside state
. We can retrieve the state.issuesDisplay
slice as one piece, and destructure the result object into multiple variables inside the component.
We now have mostly the same data variables inside the component as we did before - they're just coming from the Redux store instead of useState
hooks.
The last step is to dispatch Redux actions whenever the user does something, instead of calling the useState
setters:
Unlike typical connect
+ mapDispatch
usage, here we call dispatch()
directly, and do so by calling an action creator with the correct payload
value and passing the resulting action to dispatch
.
Let's see if this works!
If you're thinking "hey, this looks and behaves exactly like the previous example"... then that's great! That means we've correctly converted the first bit of logic to Redux so far. If you want to confirm that there's Redux logic running, try clicking the "Open in New Window" button and inspect the store in the Redux DevTools Extension.
Converting the Issues List Page
Our next task is to convert the <IssuesListPage>
component to fetch and store issues via Redux. Currently, <IssuesListPage>
is storing all data in useState
hooks, including the fetched issues. It fetches the issues by making an AJAX call in a useEffect
hook.
As mentioned at the start, there's nothing actually wrong with this! Having React components fetch and store their own data is totally fine. But, for the purposes of this tutorial, we want to see how the Redux conversion process looks.
Reviewing the Issues List Component
Here's the initial chunk of <IssuesListPage>
:
The useEffect
callback defines an outer async function fetchEverything()
and calls it immediately. This is because we can't declare the useEffect
callback itself as async. React expects that the return value from a useEffect
callback will be a cleanup function. Since all async functions return a Promise
automatically, React would see that Promise
instead, and that would prevent React from actually cleaning up correctly.
Inside, we define two more async functions to fetch issues and the open issues count, and call them both. We then wait for both functions to resolve successfully. (There's a few other ways we could have organized this logic, but this was sufficient for the example.)
Thinking in Thunks
What is a "Thunk"?
The Redux core (ie, createStore
) is completely synchronous. When you call store.dispatch()
, the store runs the root reducer, saves the return value, runs the subscriber callbacks, and returns, with no pause. By default, any asynchronicity has to happen outside of the store.
But, what if you want to have async logic interact with the store by dispatching or checking the current store state? That's where Redux middleware come in. They extend the store, and allow you to:
- Execute extra logic when any action is dispatched (such as logging the action and state)
- Pause, modify, delay, replace, or halt dispatched actions
- Write extra code that has access to
dispatch
andgetState
- Teach
dispatch
how to accept other values besides plain action objects, such as functions and promises, by intercepting them and dispatching real action objects instead
The most common Redux middleware is redux-thunk
. The word "thunk" means "a function that delays a calculation until later". In our case, adding the thunk middleware to our Redux store lets us pass functions directly to store.dispatch()
. The thunk middleware will see the function, prevent it from actually reaching the "real" store, and call our function and pass in dispatch
and getState
as arguments. So, a "thunk function" looks like this:
Inside of a thunk function, you can write any code you want. The most common usage would be fetching some data via an AJAX call, and dispatching an action to load that data into the Redux store. The async/await
syntax makes it easier to write thunks that do AJAX calls.
Normally, we don't write action objects directly in our code - we use action creator functions to make them, and use them like dispatch(addTodo())
. In the same way, we typically write "thunk action creator" functions that return the thunk functions, like:
Why Use Thunks?
You might be wondering what the point of all this is. There's a few reasons to use thunks:
- Thunks allow us to write reusable logic that interacts with a Redux store, but without needing to reference a specific store instance.
- Thunks enable us to move more complex logic outside of our components
- From a component's point of view, it doesn't care whether it's dispatching a plain action or kicking off some async logic - it just calls
dispatch(doSomething())
and moves on. - Thunks can return values like promises, allowing logic inside the component to wait for something else to finish.
For further explanations, see these articles explaining thunks in the redux-thunk
documentation.
There are many other kinds of Redux middleware that add async capabilities. The most popular are redux-saga
, which uses generator functions, and redux-observable
, which uses RxJS observables. For some comparisons, see the Redux FAQ entry on "how do I choose an async middleware?".
However, while sagas and observables are useful, most apps do not need the power and capabilities they provide. So, thunks are the default recommended approach for writing async logic with Redux.
Writing Thunks in Redux Toolkit
Writing thunk functions requires that the redux-thunk
middleware be added to the store as part of the setup process. Redux Toolkit's configureStore
function does automatically - thunk
is one of the default middleware.
However, Redux Toolkit does not currently provide any special functions or syntax for writing thunk functions. In particular, they cannot be defined as part of a createSlice()
call. You have to write them separate from the reducer logic.
In a typical Redux app, thunk action creators are usually defined in an "actions" file, alongside the plain action creators. Thunks typically dispatch plain actions, such as dispatch(dataLoaded(response.data))
.
Because we don't have separate "actions" files, it makes sense to write these thunks directly in our "slice" files. That way, they have access to the plain action creators from the slice, and it's easy to find where the thunk function lives.
Logic for Fetching Github Repo Details
Adding a Reusable Thunk Function Type
Since the thunk middleware is already set up, we don't have to do any work there. However, the TypeScript types for thunks are kind of long and confusing, and we'd normally have to repeat the same type declaration for every thunk function we write.
Before we go any further, let's add a type declaration we can reuse instead.
app/store.ts
The AppThunk
type declares that the "action" that we're using is specifically a thunk function. The thunk is customized with some additional type parameters:
- Return value: the thunk doesn't return anything
- State type for
getState
: returns ourRootState
type - "Extra argument": the thunk middleware can be customized to pass in an extra value, but we aren't doing that in this app
- Action types accepted by
dispatch
: any action whosetype
is a string.
There are many cases where you would want different type settings here, but these are probably the most common settings. This way, we can avoid repeating that same type declaration every time we write a thunk.
Adding the Repo Details Slice
Now that we have that type, we can write a slice of state for fetching details on a repo.
features/repoSearch/repoDetailsSlice.ts
The first part of this should look straightforward. We declare our slice state shape, the initial state value, and write a slice with reducers that store the open issues count or an error string, then export the action creators and reducer.
Down at the bottom, we have our first data fetching thunk. The important things to notice here are:
- The thunk is defined separately from the slice, since RTK currently has no special syntax for defining thunks as part of a slice.
- We declare the thunk action creator as an arrow function, and use the
AppThunk
type we just created. You can use either arrow functions or thefunction
keyword to write thunk functions and thunk action creators, so we could also have written this asfunction fetchIssuesCount() : AppThunk
instead. - We use the
async/await
syntax for the thunk function itself. Again, this isn't required, butasync/await
usually results in simpler code than nested Promise.then()
chains. - Inside the thunk, we dispatch the plain action creators that were generated by the
createSlice
call.
While not shown, we also add the slice reducer to our root reducer.
Async Error Handling Logic in Thunks
There is one potential flaw with the fetchIssuesCount()
thunk as written. The try/catch
block will currently catch any errors thrown
by getRepoDetails()
(such as an actual failed AJAX call), but it will also catch any errors that occur inside the dispatch of getRepoDetailsSuccess()
. In both cases, it will end up dispatch getRepoDetailsFailed()
. This may not be the desired way to handle errors, as it might show a misleading reason for what the actual error was.
There are some possible ways to restructure the code to avoid this problem. First, the await
could be switched to a standard promise chain, with separate callbacks passed in for the success and failure cases:
Or, the thunk could be rewritten to only dispatch if no errors were caught:
For sake of simplicity, we'll stick with the logic as-is for the rest of the tutorial.
Fetching Repo Details in the Issues List
Now that the repo details slice exists, we can use it in the <IssuesListPage>
component.
features/issuesList/IssuesListPage.tsx
In <IssuesListPage>
, we import the new fetchIssuesCount
thunk, and rewrite the component to read the open issues count value from the Redux store.
Inside our useEffect
, we drop the fetchIssueCount
function, and dispatch fetchIssuesCount
instead.
Logic for Fetching Issues for a Repo
Next up, we need to replace the logic for fetching a list of open issues.
features/issuesList/issuesSlice.ts
This slice is a bit longer, but it's the same basic approach as before: write the slice with reducers that handle API call results, then write thunks that do the fetching and dispatch actions with those results. The only new and interesting bits in this slice are:
- Our "start fetching" and "fetch failed" reducer logic is the same for both the single issue and multiple issue fetch cases. So, we write those functions outside the slice once, then reuse them multiple times with different names inside the
reducers
object. - The Github API returns an array of issue entries, but we want to store the data in a "normalized" structure to make it easy to look up an issue by its number. In this case, we use a plain object as a lookup table, by declaring that it is a
Record<number, Issue>
.
Fetching Issues in the Issues List
Now we can finish converting the <IssuesListPage>
component by swapping out the issues fetching logic.
Let's look at the changes.
features/issuesList/IssuesListPage.tsx
We remove the remaining useState
hooks from <IssuesListPage>
, add another useSelector
to retrieve the actual issues data from the Redux store, and construct the list of issues to render by mapping over the "current page issue IDs" array to look up each issue object by its ID.
In our useEffect
, we delete the rest of the data fetching logic that's directly in the component, and just dispatch both data fetching thunks.
This simplifies the logic in the component, but it didn't remove the work being done - it just moved it elsewhere. Again, it's not that either approach is "right" or "wrong" - it's just a question of where you want the data and the logic to live, and which approach is more maintainable for your app and situation.
Converting the Issue Details Page
The last major chunk of work left in the conversion is the <IssueDetailsPage>
component. Let's take a look at what it does.
Reviewing the Issue Details Component
Here's the current first half of <IssueDetailsPage>
, containing the state and data fetching:
It's very similar to <IssuesListPage>
. We store the current displayed Issue
, the fetched comments, and a potential error. We have useEffect
hooks that fetch the current issue by its ID, and fetch the comments whenever the issue changes.
Fetching the Current Issue
We conveniently already have the Redux logic for fetching a single issue - we wrote that already as part of issuesSlice.ts
. So, we can immediately jump straight to using that here in <IssueDetailsPage>
.
features/issueDetails/IssueDetailsPage.tsx
We continue the usual pattern. We drop the existing useState
hooks, pull in useDispatch
and the necessary state via useSelector
, and dispatch the fetchIssue
thunk to fetch data.
Interestingly, there's actually a bit of a change in behavior here. The original React code was storing the fetched issues in <IssuesListPage>
, and <IssueDetailsPage>
was always having to do a separate fetch for its own issue. Because we're now storing issues in the Redux store, most of the time the listed issue should be already cached, and we don't even need to fetch it. Now, it's totally possible to do something similar with just React - all we'd have to do is pass the issue down from the parent component. Still, having that data in Redux makes it easier to do the caching.
(As an interesting side note: the original code always caused the page to jump back to the top, because the issue didn't exist during the first render, so there was no content. If the issue does exist and we render it right away, the page may retain the scroll position from the issues list, so we have to enforce scrolling back to the top.)
Logic for Fetching Comments
We have one more slice left to write - we need to fetch and store comments for the current issue.
features/issueDetails/commentsSlice.ts
The slice should look pretty familiar at this point. Our main bit of state is a lookup table of comments keyed by an issue ID. After the slice, we add a thunk to fetch the comments for a given issue, and dispatch the action to save the resulting array in the slice.
Fetching the Issue Comments
The final step is to swap the comments fetching logic in <IssueDetailsPage>
.
features/issueDetails/IssueDetailsPage.tsx
We add another useSelector
hook to pull out the current comments data. In this case, we need three different pieces: the loading flag, a potential error, and the actual comments array for this issue.
However, this leads to a performance problem. Every time this selector runs, it returns a new object: {commentsLoading, commentsError, comments}
. Unlike connect
, useSelector
relies on reference equality by default. So, returning a new object will cause this component to rerender every time an action is dispatched, even if the comments are the same!
There's a few ways to fix this:
- We could write those as separate
useSelector
calls - We could use a memoized selector, such as
createSelector
from Reselect - We can use the React-Redux
shallowEqual
function to compare the results, so that the re-render only happens if the object's contents have changed.
In this case, we'll add shallowEqual
as the comparison function for useSelector
.
Summary
And with that, we're done! The entire Github Issues app should now be fetching its data via thunks, storing the data in Redux, and interacting with the store via React-Redux hooks. We have Typescript types for our Github API calls, the API types are being used for the Redux state slices, and the store state types are being used in our React components.
There's more that could be done to add more type safety if we wanted (like trying to constrain which possible action types can be passed to dispatch
), but this gives us a reasonable "80% solution" without too much extra effort.
Hopefully you now have a solid understanding of how Redux Toolkit looks in a real world application.
Let's wrap this up with one more look at the complete source code and the running app:
Now, go out there and build something cool!