0

0

React应用中Axios异步数据顺序渲染问题解析与优化

心靈之曲

心靈之曲

发布时间:2025-08-08 21:04:01

|

362人浏览过

|

来源于php中文网

原创

React应用中Axios异步数据顺序渲染问题解析与优化

本文旨在解决React应用中因Axios异步请求和状态更新机制不当导致的UI元素渲染顺序错乱问题。通过深入分析错误的异步处理模式,如在循环中进行非同步状态更新,并提出使用async/await语法结合Promise.all进行批量数据获取和一次性状态更新的优化方案。此方法能确保数据按预期顺序加载并渲染,提高应用稳定性和用户体验。

1. 问题背景:异步数据获取与渲染顺序错乱

在react应用中,当我们需要从api获取大量数据并进行渲染时,常常会遇到数据加载顺序与预期不符的问题。这通常发生在以下场景:首先获取一个列表,然后对列表中的每个项目进行单独的详细信息查询。如果这些详细信息查询是异步的,并且每次查询完成后立即更新组件状态,那么最终渲染的顺序将取决于网络请求完成的先后,而非原始列表的顺序,导致ui元素显示混乱。

以上述Pokédex应用为例,其核心逻辑是:

  1. 获取前151个宝可梦的列表(包含名称和URL)。
  2. 遍历这个列表,对每个宝可梦发起单独的请求,获取其详细信息。
  3. 将获取到的详细信息添加到pokedex状态中。

原代码中,setPokedex操作在mapPokemon函数内部被多次调用,且每次调用都是在各自的异步请求完成后立即执行。由于网络请求的异步性,这些请求的完成顺序是不可预测的。这意味着,即使原始列表是按ID排序的,但由于网络延迟,ID为50的宝可梦可能比ID为10的宝可梦更早完成数据获取并更新状态,从而导致最终渲染顺序错乱。

2. 原始代码分析及潜在问题

让我们详细审视原始代码中的关键部分:

// collectPokemon 函数
const collectPokemon = async (limit: number) => {
  axiosInstance.get(`pokemon?limit=${limit}`).then((res) => {
    const data = res.data.results; // data 是一个包含 {name, url} 的数组

    // 错误用法:Promise.all(data) - data 并非 Promise 数组
    Promise.all(data).then((results) => {
      results.map((pkmn) => {
        mapPokemon(pkmn.name); // 调用 mapPokemon,但其返回的 Promise 未被收集
      });
    });
  });
};

// mapPokemon 函数
const mapPokemon = async (name: string) => {
  axiosInstance.get(`pokemon/${name}`).then((res) => {
    const data: PokemonType = res.data;
    // 问题所在:在循环中多次更新状态
    setPokedex((currentList) => [...currentList, data]);
  });
};

存在以下主要问题:

  1. Promise.all(data) 的误用:data (res.data.results) 是一个普通的 JavaScript 对象数组,而不是 Promise 数组。Promise.all期望接收一个 Promise 数组,它会等待所有 Promise 都解决后才执行其 .then 回调。在这里,它并没有起到等待所有详细宝可梦数据加载完成的作用。
  2. mapPokemon 的返回值未被利用:在 results.map((pkmn) => mapPokemon(pkmn.name)); 这行代码中,mapPokemon 函数确实被调用了,并且它返回一个 Promise。但是,这个 map 操作的结果(一个 Promise 数组)并没有被收集起来,也没有被 Promise.all 等待。这意味着外部的 collectPokemon 函数并没有等待所有 mapPokemon 调用完成。
  3. 在循环中多次更新状态 (setPokedex):这是导致渲染顺序错乱的根本原因。每次 mapPokemon 函数中的 axiosInstance.get 请求成功后,都会立即调用 setPokedex 来更新 pokedex 状态。由于网络请求的异步性和不可预测的完成顺序,setPokedex 被调用的顺序与宝可梦的原始ID顺序不一致,从而导致最终渲染的列表顺序混乱。频繁地更新状态也可能带来不必要的渲染开销。

3. 解决方案:使用 async/await 和 Promise.all 进行批量更新

解决此问题的关键在于:

  1. 确保所有详细数据请求都完成后,再进行一次性状态更新。
  2. 利用 async/await 语法使异步代码更具可读性。
  3. 正确使用 Promise.all 来等待一组 Promise 全部解决。

以下是优化后的代码结构:

VidAU
VidAU

VidAU AI 是一款AI驱动的数字人视频创作平台,旨在简化视频内容创作流程

下载
import { useEffect, useState } from "react";
import "./App.css";
import axios from "axios";
import { PokemonType } from "./models/pokemonType";
import Pokemon from "./components/Pokemon";

function App() {
  const [pokedex, setPokedex] = useState([]);
  const limit = 151;

  const axiosInstance = axios.create({
    baseURL: "https://pokeapi.co/api/v2/",
  });

  useEffect(() => {
    // 在组件挂载时调用 collectPokemon
    collectPokemon(limit);
  }, []); // 空数组依赖表示只在挂载时运行一次

  /**
   * 收集指定数量的宝可梦数据
   * @param limit 要收集的宝可梦数量
   */
  const collectPokemon = async (limit: number) => {
    try {
      // 1. 获取宝可梦列表(包含名称和URL)
      const res = await axiosInstance.get(`pokemon?limit=${limit}`);
      const results = res.data.results; // results 是一个包含 {name, url} 的数组

      // 2. 为每个宝可梦发起详细信息请求,并收集所有这些请求的 Promise
      // mapPokemon 现在应该返回 Promise
      const promises = results.map((pkmn: { name: string; url: string }) =>
        mapPokemon(pkmn.name)
      );

      // 3. 使用 Promise.all 等待所有详细信息请求完成
      // pokemons 将是一个包含所有 PokemonType 对象的有序数组
      const pokemons = await Promise.all(promises);

      // 4. 一次性更新状态,确保数据顺序正确
      setPokedex(pokemons);
    } catch (error) {
      console.error("Failed to collect Pokemon data:", error);
      // 可以在这里处理错误,例如显示错误消息给用户
    }
  };

  /**
   * 获取单个宝可梦的详细信息
   * @param name 宝可梦的名称
   * @returns 返回一个 Promise,解析为 PokemonType 对象
   */
  const mapPokemon = async (name: string): Promise => {
    const res = await axiosInstance.get(`pokemon/${name}`);
    return res.data; // 直接返回获取到的数据
  };

  console.log(pokedex);

  return (
    <>
      

Pokédex App

{pokedex.map((mon: PokemonType) => ( ))}
); } export default App;

Pokemon.tsx (保持不变,但需注意 key 的使用)

Pokemon.tsx 组件的代码在此问题中不是直接原因,但为了完整性,仍然需要确保其 key 属性的正确使用。在 React 中,列表渲染时 key 属性非常重要,它帮助 React 识别列表中哪些项发生了变化、被添加或被删除。理想情况下,key 应该是数据项的稳定唯一标识符,如 mon.id。

import { PokemonType } from "../models/pokemonType";

const Pokemon = (props: PokemonType) => {
  const { name, id, types } = props;
  const paddedIndex = ("000" + (id ? id : id)).slice(-3);

  return (
    
#{paddedIndex}
@@##@@
{name}
{types?.map((item) => ( // 这里的 key 也应该是一个稳定值

{item.type?.name}

))}
); }; export default Pokemon;

关于 key 属性的注意事项: 在 Pokemon.tsx 中,img 标签上的 key={id} 是多余的,因为 Pokemon 组件本身已经在 App.tsx 的 map 方法中使用了 key={mon.id}。更重要的是,在 types.map 循环中,key={Math.random()} 是一个严重错误。Math.random() 每次渲染都会生成一个不同的值,这会导致 React 无法正确追踪列表项的变化,从而可能引发性能问题或不必要的重新渲染。应该使用类型名称或其他稳定唯一标识符作为 key,例如 key={item.type?.name}。

4. 关键点总结与最佳实践

  • async/await 语法:它使得异步代码的编写和阅读更接近同步代码,避免了 .then() 回调的层层嵌套(回调地狱),提高了代码的可维护性。
  • Promise.all() 的正确使用:当需要并行执行多个异步操作,并等待所有操作都完成后再进行下一步处理时,Promise.all() 是最佳选择。它接收一个 Promise 数组,并返回一个新的 Promise,该 Promise 在所有输入的 Promise 都成功解决后解决,其结果是一个包含所有 Promise 解决值的数组,且顺序与输入 Promise 的顺序一致。
  • 批量状态更新:避免在循环或异步回调中频繁地调用 setPokedex 等状态更新函数。收集所有数据,然后进行一次性状态更新,这不仅能保证数据顺序的正确性,还能减少不必要的组件渲染,从而优化应用性能。
  • 错误处理:在异步操作中加入 try...catch 块是良好的实践,可以捕获网络请求失败等异常,提高应用的健壮性。
  • useEffect 依赖项:确保 useEffect 的依赖项数组正确设置。在此例中,[] 表示效果只在组件挂载时运行一次,这对于初始化数据获取是合适的。
  • key 属性的重要性:在 React 列表渲染中,为每个列表项提供一个稳定、唯一的 key 至关重要。它帮助 React 有效地识别和更新列表中的元素。避免使用 Math.random() 或数组索引作为 key,除非列表项的顺序和内容永不改变。

通过采纳上述优化方案,Pokédex应用将能够确保所有宝可梦数据按照其原始ID顺序正确加载并显示,提供稳定且一致的用户体验。

{name}

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

557

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

395

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

756

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

478

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

474

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

1051

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

659

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

554

2023.09.20

菜鸟裹裹入口以及教程汇总
菜鸟裹裹入口以及教程汇总

本专题整合了菜鸟裹裹入口地址及教程分享,阅读专题下面的文章了解更多详细内容。

0

2026.01.22

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3万人学习

CSS教程
CSS教程

共754课时 | 22.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号