0

我是使用 redux 的新手,我正在尝试将 redux-promise 设置为中间件。我有这种情况,我似乎无法开始工作(当我只是想在没有链接的情况下进行一次异步调用时,事情对我有用)

假设我有两个 API 调用:

1) getItem(someId) -> {attr1: something, attr2: something, tagIds: [...]}
2) getTags() -> [{someTagObject1}, {someTagObject2}]

我需要调用第一个,并获取一个项目,然后获取所有标签,然后返回一个包含项目和与该项目相关的标签的对象。

现在,我的动作创建者是这样的:

export function fetchTagsForItem(id = null, params = new Map()) {
    return {
        type: FETCH_ITEM_INFO,
        payload: getItem(...) // some axios call
            .then(item => getTags() // gets all tags 
                .then(tags => toItemDetails(tags.data, item.data)))
    }
}

我在toItemDetails中有一个console.log,我可以看到当调用完成后,我们最终进入toItemDetails并得到正确的信息。但是,看起来我们在调用完成之前就到达了减速器,而我只是从减速器获得了一个未定义的有效负载(它不会再试一次)。对于这种情况,reducer 只是试图返回 action.payload。

我知道链接调用不是很好,但我至少希望看到它工作。这是可以通过 redux-promise 完成的吗?如果没有,任何有关如何实现此功能的示例将不胜感激!

4

1 回答 1

0

我用占位符函数填写了您丢失的代码,它对我有用 - 我的有效负载最终包含一个解析为返回值的承诺toItemDetails。因此,也许这是您未在此处包含的代码中的某些内容。

function getItem(id) {
  return Promise.resolve({
    attr1: 'hello',
    data: 'data inside item',
    tagIds: [1, 3, 5]
  });
}

function getTags(tagIds) {
  return Promise.resolve({ data: 'abc' });
}

function toItemDetails(tagData, itemData) {
  return { itemDetails: { tagData, itemData } };
}

function fetchTagsForItem(id = null) {
  let itemFromAxios;
  return {
    type: 'FETCH_ITEM_INFO',
    payload: getItem(id)
      .then(item => {
        itemFromAxios = item;
        return getTags(item.tagIds);
      })
      .then(tags => toItemDetails(tags.data, itemFromAxios.data))
  };
}


const action = fetchTagsForItem(1);
action.payload.then(result => {
  console.log(`result:  ${JSON.stringify(result)}`);
});

输出:

result:  {"itemDetails":{"tagData":"abc","itemData":"data inside item"}}

为了item在第二步中进行访问,您需要将其存储在一个在函数作用域中声明的变量中fetchTagsForItem,因为这两个.thens 本质上是兄弟姐妹:两者都可以访问封闭作用域,但第二次调用.thenwon' t 可以访问在第一个声明中声明的变量。

关注点分离

创建您发送到 Redux 的操作的代码也会进行多次 Axios 调用并处理返回的数据。这使得阅读和理解变得更加复杂,并且使处理 Axios 调用中的错误等事情变得更加困难。我建议把事情分开。一种选择:

  • 将任何调用axios的代码放在自己的函数中
  • 设置payload为该函数的返回值。
  • 将该函数以及所有其他调用 Axios 的函数移动到一个单独的文件(或一组文件)中。该文件成为您的 API 客户端。

这看起来像:

// apiclient.js
const BASE_URL = 'https://yourapiserver.com/';
const makeUrl = (relativeUrl) => BASE_URL + relativeUrl; 
function getItemById(id) {
  return axios.get(makeUrl(GET_ITEM_URL) + id);
}

function fetchTagsForItemWithId(id) {
...
}

// Other client calls and helper funcs here

export default {
  fetchTagsForItemWithId
};

您的操作文件:

// items-actions.js
import ApiClient from './api-client';
function fetchItemTags(id) {
  const itemInfoPromise = ApiClient.fetchTagsForItemWithId(id);
  return {
    type: 'FETCH_ITEM_INFO',
    payload: itemInfoPromise 
  };
}
于 2018-01-21T03:35:29.117 回答