使用异步action

redux除了可以执行同步action外,还可以执行异步action.

标准的做法是使用 Redux Thunk 中间件。要引入 redux-thunk 这个专门的库才能使用。我们 后面 会介绍 middleware 大体上是如何工作的;目前,你只需要知道一个要点:通过使用指定的 middleware,action 创建函数除了返回 action 对象外还可以返回函数。这时,这个 action 创建函数就成为了 thunk不是只有返回了异步函数才能成为chunk
当 action 创建函数返回函数时,这个函数会被 Redux Thunk middleware 执行。这个函数并不需要保持纯净;它还可以带有副作用,包括执行异步 API 请求。这个函数还可以 dispatch action,就像 dispatch 前面定义的同步 action 一样。
我们仍可以在 actions.js 里定义这些特殊的 thunk action 创建函数。

  1. export function getCurrentTime() {
  2. return function fetchCurrentTime(dispatch) {
  3. dispatch(pending());
  4. return fetch("http://127.0.0.1:8080/api/common/currentTime")
  5. .then(
  6. response => response.json(),
  7. error => console.log("error", error)
  8. )
  9. .then(res => {
  10. if(res.msg === "ok") {
  11. dispatch(setCurrentTime({
  12. currentTime2: res.data.currentTime,
  13. status: "success",
  14. msg: "ok",
  15. }))
  16. } else {
  17. dispatch(fail())
  18. }
  19. })
  20. }
  21. }

在dispatch中使用中间件需要引入Redux Thunk middleware:

  1. import thunk from "redux-thunk";
  2. const { createStore, applyMiddleware } = require("redux");
  3. const { timeTool } = require("./reducer");
  4. let store = createStore(timeTool, applyMiddleware(thunk));
  5. export default store;

再次dispatch

thunk 的一个优点是它的结果可以再次被 dispatch:

  1. import fetch from 'cross-fetch'
  2. export const REQUEST_POSTS = 'REQUEST_POSTS'
  3. function requestPosts(subreddit) {
  4. return {
  5. type: REQUEST_POSTS,
  6. subreddit
  7. }
  8. }
  9. export const RECEIVE_POSTS = 'RECEIVE_POSTS'
  10. function receivePosts(subreddit, json) {
  11. return {
  12. type: RECEIVE_POSTS,
  13. subreddit,
  14. posts: json.data.children.map(child => child.data),
  15. receivedAt: Date.now()
  16. }
  17. }
  18. export const INVALIDATE_SUBREDDIT = 'INVALIDATE_SUBREDDIT'
  19. export function invalidateSubreddit(subreddit) {
  20. return {
  21. type: INVALIDATE_SUBREDDIT,
  22. subreddit
  23. }
  24. }
  25. function fetchPosts(subreddit) {
  26. return dispatch => {
  27. dispatch(requestPosts(subreddit))
  28. return fetch(`http://www.reddit.com/r/${subreddit}.json`)
  29. .then(response => response.json())
  30. .then(json => dispatch(receivePosts(subreddit, json)))
  31. }
  32. }
  33. function shouldFetchPosts(state, subreddit) {
  34. const posts = state.postsBySubreddit[subreddit]
  35. if (!posts) {
  36. return true
  37. } else if (posts.isFetching) {
  38. return false
  39. } else {
  40. return posts.didInvalidate
  41. }
  42. }
  43. export function fetchPostsIfNeeded(subreddit) {
  44. // 注意这个函数也接收了 getState() 方法
  45. // 它让你选择接下来 dispatch 什么。
  46. // 当缓存的值是可用时,
  47. // 减少网络请求很有用。
  48. return (dispatch, getState) => {
  49. if (shouldFetchPosts(getState(), subreddit)) {
  50. // 在 thunk 里 dispatch 另一个 thunk!
  51. return dispatch(fetchPosts(subreddit))
  52. } else {
  53. // 告诉调用代码不需要再等待。
  54. return Promise.resolve()
  55. }
  56. }
  57. }

thunk middleware并不是处理异步action的唯一方式:

你也可以先尝试一些不同做法,选择喜欢的,并使用下去,不论有没有使用到 middleware 都行。

异步数据流