Hallo zusammen!
Ich habe an einem leichten React-Hook gearbeitet, den ich useAsync nannte und der einige der wesentlichen Funktionen von React Query nachahmt (wie Abrufen, Caching, Wiederholungsversuche usw.), aber in einer kompakteren, leicht anpassbares Paket. Nachfolgend finden Sie eine kurze Aufschlüsselung der internen Funktionsweise unter Bezugnahme auf die relevanten Codeabschnitte. Wenn Sie den gesamten Code sehen möchten, gehen Sie zum Repo:
Vollständiger Quellcode auf GitHub.
Der Hook ist auch auf npm als api-refetch verfügbar.
Während React Query und SWR beide großartige Bibliotheken sind, wollte ich aus mehreren Gründen einen praktischeren Ansatz:
Leichter Fußabdruck
Obwohl React Query und SWR funktionsreich sind, können sie relativ groß sein (React Query ~2,2 MB, SWR ~620 kB). API-Refetch hat eine Größe von etwa 250 kB und ist somit ideal für kleinere Apps, bei denen die Bundle-Größe eine große Rolle spielt. Dieser Hook sollte als Abhängigkeit einer anderen Bibliothek (Intlayer) installiert werden. Daher war die Größe der Lösung ein wichtiger Gesichtspunkt.
Einfach anzupassen und zu optimieren
Ich brauchte einige spezifische Funktionen – wie Speichern/Abrufen von Daten aus dem lokalen Speicher und Verwaltung paralleler Anfragen mit einem unkomplizierten Ansatz.
Durch Klonen des Repos oder Kopieren des Codes direkt in Ihr Projekt können Sie alle unerwünschten Funktionen entfernen und nur das behalten, was Sie benötigen. Dies reduziert nicht nur die Bündelgröße, sondern minimiert auch unnötige Neu-Renderings und -Vergrößerungen, sodass Sie eine schlankere, leistungsfähigere Lösung erhalten, die auf Ihre spezifischen Anforderungen zugeschnitten ist.
Kein Anbieter erforderlich
Ich wollte Context Provider vermeiden, um den Hook global zu machen und seine Verwendung so einfach wie möglich zu halten. Also habe ich eine Version des Hakens basierend auf einem Zustand-Shop erstellt (siehe Beispiel unten).
Lernübung
Der Aufbau einer asynchronen Bibliothek von Grund auf ist eine hervorragende Möglichkeit, Parallelität, Caching und die internen Aspekte der Zustandsverwaltung zu verstehen.
Kurz gesagt, das Erstellen meines eigenen Hakens war eine Chance, genau die Funktionen zu verfeinern, die ich brauche (und diejenigen zu überspringen, die ich nicht brauche) und gleichzeitig die Bibliothek klein und leicht verständlich zu halten.
Der React-Hook verwaltet:
Im Folgenden finden Sie die wichtigsten Punkte in api-refetch und kurze Verweise auf die relevanten Teile des Codes in useAsync.tsx.
// This map stores any in-progress Promise to avoid sending parallel requests // for the same resource across multiple components. const pendingPromises = new Map(); const fetch: T = async (...args) => { // Check if a request with the same key + args is already running if (pendingPromises.has(keyWithArgs)) { return pendingPromises.get(keyWithArgs); } // Otherwise, store a new Promise and execute const promise = (async () => { setQueryState(keyWithArgs, { isLoading: true }); // ...perform fetch here... })(); // Keep it in the map until it resolves or rejects pendingPromises.set(keyWithArgs, promise); return await promise; };
// Handle periodic revalidation if caching is enabled useEffect( () => { if (!revalidationEnabled || revalidateTime <= 0) return; // Revalidation is disabled if (!isEnabled || !enabled) return; // Hook is disabled if (isLoading) return; // Fetch is already in progress if (!isSuccess || !fetchedDateTime) return; // Should retry either of revalidate if (!(cacheEnabled || storeEnabled)) return; // Useless to revalidate if caching is disabled const timeout = setTimeout(() => { fetch(...storedArgsRef.current); }, revalidateTime); return () => clearTimeout(timeout); }, [ /* dependencies */ ] );
useEffect( () => { const isRetryEnabled = errorCount > 0 && retryLimit > 0; const isRetryLimitReached = errorCount > retryLimit; if (!isEnabled || !enabled) return; // Hook is disabled if (!isRetryEnabled) return; // Retry is disabled if (isRetryLimitReached) return; // Retry limit has been reached if (!(cacheEnabled || storeEnabled)) return; // Useless to retry if caching is disabled if (isLoading) return; // Fetch is already in progress if (isSuccess) return; // Hook has already fetched successfully const timeout = setTimeout(() => { fetch(...storedArgsRef.current); }, retryTime); return () => clearTimeout(timeout); }, [ /* dependencies */ ] );
// Auto-fetch data on hook mount if autoFetch is true useEffect( () => { if (!autoFetch) return; // Auto-fetch is disabled if (!isEnabled || !enabled) return; // Hook is disabled if (isFetched && !isInvalidated) return; // Hook have already fetched or invalidated if (isLoading) return; // Fetch is already in progress fetch(...storedArgsRef.current); }, [ /* dependencies */ ] );
Sehen Sie sich hier den vollständigen Code an, der lokale Speicherlogik, Abfrageungültigmachung und mehr enthält:
Probieren Sie es einfach aus, melden Sie Probleme oder leisten Sie einen Beitrag, wenn Sie interessiert sind. Wir freuen uns über jedes Feedback!
Kopieren Sie den Code oder codieren Sie das (Repo)[https://github.com/aymericzip/api-refetch]
Oder
// This map stores any in-progress Promise to avoid sending parallel requests // for the same resource across multiple components. const pendingPromises = new Map(); const fetch: T = async (...args) => { // Check if a request with the same key + args is already running if (pendingPromises.has(keyWithArgs)) { return pendingPromises.get(keyWithArgs); } // Otherwise, store a new Promise and execute const promise = (async () => { setQueryState(keyWithArgs, { isLoading: true }); // ...perform fetch here... })(); // Keep it in the map until it resolves or rejects pendingPromises.set(keyWithArgs, promise); return await promise; };
// Handle periodic revalidation if caching is enabled useEffect( () => { if (!revalidationEnabled || revalidateTime <= 0) return; // Revalidation is disabled if (!isEnabled || !enabled) return; // Hook is disabled if (isLoading) return; // Fetch is already in progress if (!isSuccess || !fetchedDateTime) return; // Should retry either of revalidate if (!(cacheEnabled || storeEnabled)) return; // Useless to revalidate if caching is disabled const timeout = setTimeout(() => { fetch(...storedArgsRef.current); }, revalidateTime); return () => clearTimeout(timeout); }, [ /* dependencies */ ] );
Das ist es! Probieren Sie es aus und lassen Sie mich wissen, wie es läuft. Feedback, Fragen oder Beiträge sind auf GitHub mehr als willkommen.
GitHub: api-refetch
Viel Spaß beim Codieren!
Das obige ist der detaillierte Inhalt vonPräsentation eines leichtgewichtigen Hooks für den asynchronen Datenabruf und das Caching in React. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!