
在react应用中,当我们需要从api获取大量数据并进行渲染时,常常会遇到数据加载顺序与预期不符的问题。这通常发生在以下场景:首先获取一个列表,然后对列表中的每个项目进行单独的详细信息查询。如果这些详细信息查询是异步的,并且每次查询完成后立即更新组件状态,那么最终渲染的顺序将取决于网络请求完成的先后,而非原始列表的顺序,导致ui元素显示混乱。
以上述Pokédex应用为例,其核心逻辑是:
原代码中,setPokedex操作在mapPokemon函数内部被多次调用,且每次调用都是在各自的异步请求完成后立即执行。由于网络请求的异步性,这些请求的完成顺序是不可预测的。这意味着,即使原始列表是按ID排序的,但由于网络延迟,ID为50的宝可梦可能比ID为10的宝可梦更早完成数据获取并更新状态,从而导致最终渲染顺序错乱。
让我们详细审视原始代码中的关键部分:
// 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]);
});
};存在以下主要问题:
解决此问题的关键在于:
以下是优化后的代码结构:
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<PokemonType[] | []>([]);
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<PokemonType>
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<PokemonType> => {
const res = await axiosInstance.get(`pokemon/${name}`);
return res.data; // 直接返回获取到的数据
};
console.log(pokedex);
return (
<>
<h1>Pokédex App</h1>
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-10 m-10">
{pokedex.map((mon: PokemonType) => (
<Pokemon key={mon.id} name={mon.name} id={mon.id} types={mon.types} />
))}
</div>
</>
);
}
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 (
<div className="bg-gray-600 rounded-2xl p-2 transform h-180 min-w-250 transition duration-500 hover:scale-180 hover:drop-shadow-[0_10px_10px_rgba(0,0,0,.5)] border-gray-950 border-4 overflow-clip">
<div className="bg-gray-900 text-white rounded-xl p-2 w-20 text-center absolute top-2 right-2 font-pokemonGB border-white border-2">
#{paddedIndex}
</div>
<img
// key 属性应该放在列表渲染的顶级元素上,而不是 img 标签上
// 这里 img 标签的 key 是多余的,但不会引起错误
key={id}
className="scale-125 relative"
width={150}
height={150}
alt={name}
src={`https://assets.pokemon.com/assets/cms2/img/pokedex/detail/${paddedIndex}.png`}
/>
<div className="font-pokemonGB text-black bg-white w-auto text-sm text-center capitalize rounded-tl-2xl p-2 absolute right-0 bottom-0 slant-x-[30deg] border-gray-950 border-4 -m-1">
{name}
</div>
<div
// key={Math.random()} 在这里是错误的用法,每次渲染都会生成新的 key
// 应该使用 item.type?.name 或 item.type?.id 作为 key
key={Math.random()}
className=" flex flex-row fixed right-2 top-16 space-x-3"
>
{types?.map((item) => (
// 这里的 key 也应该是一个稳定值
<div key={item.type?.name} className="bg-red-500 rounded-lg p-2 capitalize border-gray-950 border-2 ">
<p>{item.type?.name}</p>
</div>
))}
</div>
</div>
);
};
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}。
通过采纳上述优化方案,Pokédex应用将能够确保所有宝可梦数据按照其原始ID顺序正确加载并显示,提供稳定且一致的用户体验。
以上就是React应用中Axios异步数据顺序渲染问题解析与优化的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号