Data fetching to pobieranie danych z serwera lub API oraz obsługa cache, błędów, retry i stanu ładowania., czyli sztuka sensownego pobierania danych, to bodajże najstarszy problem w
świecie Reacta i jednocześnie decyzja, którą architekci systemów potrafią
najmocniej zaniedbać. Napisanie podstawowego useEffect wraz z fetch jest
dziecinnie proste. Jednak dopisanie do niego logiki, która poprawnie obsłuży
cache, usunie zdublowane zapytania, bezbłędnie zadba o powtórzenia w razie
błędów (retry) i elegancko wybroni się przed nadpisywaniem starych wyników (race
conditions)... to już temat na kilka dobrych miesięcy ciężkiej pracy całego
zespołu.
Mamy 2026 rok i na stole wylądowały w zasadzie trzy dominujące rozwiązania: potężny kombajn TanStack Query, zwinny i lekki pakiet SWR od Vercela oraz uparty klasyk useEffect (stosowany głównie przez zespoły cierpiące na alergię na dodawanie nowych bibliotek do pliku package.json). W tym artykule zderzymy ze sobą całą trójkę i sprawdzimy, jak w to wszystko wpisują się rewelacyjne Server Components.
Hierarchia środowiska – ścieżka podejmowania decyzji na skróty
Nie masz czasu na długie lektury? Oto ściągawka w 30 sekund:
- Server Components (w Next.js lub Remix) — Twoja pierwsza linia frontu. Jeśli coś da się zaciągnąć i wrzucić do HTML-a bezpośrednio z poziomu serwera, zrób to tam. Nie martwisz się o logikę cache'owania, unikasz wpychania JavaScriptu do przeglądarki.
- TanStack Query — Podstawowe działo, jeśli jednak potrzebujesz uderzyć po dane z poziomu frontendu (klienta). Potężne, stabilne i przeładowane niezbędnymi funkcjami.
- SWR — Gdy budujesz coś bardzo minimalistycznego i potrzebujesz jedynie leciutkiej odskoczni pod podstawowe odpytywania.
- Opcja z
useEffect+fetch— Zakazane słowo przy twardej architekturze korporacyjnej. Trzymaj to z dala od środowisk produkcyjnych.
Dlaczego useEffect potrafi mocno podciąć skrzydła?
Zacznijmy w ogóle od tego, na jakich minach kładzie nas klasyczne, podręcznikowe podejście:
Powyższy blok kodu technicznie... działa. Problem polega na tym, że skrywa mnóstwo architektonicznych potknięć:
1. Wycieki na zasobach bazy (Brak cache'owania). Wyobraź sobie, że wchodzisz w zakładkę /users/123, potem skaczesz pod /users/456 by za moment znów wrócić na stare /users/123. Przeglądarka zaciągnie kod od zera za każdym jednym razem. Choć rekordy w bazie się nie zmieniły – Ty niepotrzebnie płacisz za to w rachunkach do serwera.
2. Marnowanie łącza (Brak deduplikacji). Jeżeli na jednej stronie pięć różnych małych komponentów zechce odczytać dane z /api/users/123, "goły" fetch potulnie wyśle w świat pięć osobnych requestów. Kompletny absurd i to bardzo powszechny.
3. "Kto pierwszy ten lepszy" (Race conditions). Nawet z opcją zmyślnej blokady (flaga cancelled), jeśli szybko przebierasz między podstronami, asynchroniczne odpowiedzi mogą spłynąć do komponentu w całkowicie złej kolejności. Twój kod wysypie starą odpowiedź, chociaż przed momentem pytałeś już o nową.
4. Panika przy błędach (Brak retry). Padło Ci lokalne połączenie na ułamek sekundy w pociągu? Koniec. Użytkownik dostaje brzydki Error i jeśli sam nie wciśnie klawisza f5 (odśwież), utknie tam na amen.
5. Złe doświadczenia w interfejsie (Brak odświeżeń w tle). Odchodzisz od biurka, w międzyczasie Twoi znajomi dopisali wpisy w grupie. Wracasz za kwadrans, ale strona twardo wisi na tym co widziała przed wyjściem. Bez odświeżania na tło nie wiesz nawet, że informacje są przestarzałe.
6. Frustrujący wskaźnik ładowania (Brak Optymistycznych Aktualizacji). Zostawiasz komentarz, po czym wlepiasz wzrok w kręcące się kółko, modląc się, by proces przebiegł pomyślnie. Nowoczesne systemy pokazują Twój komentarz od razu, nie każąc Ci czekać na zielone światło od API.
Widzisz problem? Oczywiście, jako dzielny rzemieślnik mógłbyś to obudować samemu w kodzie, ale z ręką na sercu — napisanie tych warstw od nowa to tysiące linii kodu i setki potknięć (bugów). Po co wyważać otwarte drzwi, skoro inni mądrzy specjaliści od bibliotek zrobili to świetnie przed nami?
TanStack Query — Prawdziwy hegemon roku 2026
TanStack Query (w starych dziejach używany pod szyldem React Query) to paczka, która powyższe problemy kompresuje do śmiesznie prostych dziesięciu linijek kodu.
Tylko tyle. Co w pakiecie dostarcza Ci ten kod?
- Genialnie prosty bufor - rozbija cache na podstawie podanego klucza (czyli np.
['user', userId]). Każdy kolejny komponent podpięty pod ten klucz, bez pardonu doczepi się do wyniku i daruje sobie szukanie danych na świeżo w bazie. - Magiczna Deduplikacja - jak już wcześniej wspomniałem, wszystkie prośby równoległe zostają zebrane i wysłane jednym zgrabnym "kurierem" (promisem).
- Walka do końca (Retry) - Domyślnie skrypt walczy jeszcze trzykrotnie, po równych odstępach odrzucenia (exponential backoff) zanim definitywnie "rzuci ręcznikiem".
- Natychmiastowe aktualizacje dla wracających - Zmienisz kartę by sprawdzić maila? TanStack na moment przed Twoim powrotem odpyta API, żeby zaserwować Ci na ekran najbardziej "świeżą" treść.
- Rewalidacja Stale-while - Pokaże to, co do tej pory uzbierał w buforze w błyskawicznym trybie, żeby tylko w tle móc doładować i uaktualnić wartości.
Przechodzimy do małego wdrożenia (Setup)
Teraz mała owijka dla reszty aplikacji:
Mała podpowiedź jeśli pracujesz nad stosem od Next.js z App Routerem — tam upewnij się, żeby Twój wbudowany "klient do zapytań" stał się funkcją opartą per każde wejście a nie o jeden instancyjny, wspólny "singleton"! Dokumentacja na oficjalnych stronach przeprowadzi Cię przez to wzorowo.
Wbudowane, agresywne domyślne zasady od twórców (Które trzeba znać by nie płakać!)
TanStack nie "prosi". TanStack narzuca swoje żelazne i niezwykle czułe ramy na cały framework. Nieznajomość tych domyślnych funkcji bywa niezwykle myląca dla programistów:
| Ustawienie Startowe | Za co opowiada u podstaw? | Kiedy powinieneś interweniować i je przesterować? |
|---|---|---|
staleTime: 0 | W tej paczce wszystko, dosłownie "natychmiast" bywa traktowane za przeterminowane (stale). Odświeżenia i pule pobrań w tle odpalają w sekundę po przeładowaniach widoków! | Odmierz spokojnie np 60 do 120 sekund czasu dla mało krytycznych miejsc pod odświeżanie serwerowe by portfele u inwestora nie poszły z ogniem. |
gcTime: 5 * 60 * 1000 | Gdyby skrypt o tobie kompletnie zapomniał (bo przeszedłeś w panel na całkowicie inny model komponentowy) to po pełnych pięciu minutach zapas bufora poleci "w kosz". | Wyjdź w wyższe ramy, kiedy użytkownicy Twojego klienta notorycznie lubią wracać do wielkich rozstrzelanych i wczytanych wcześniej wielkich raportów. |
retry: 3 | Każde polegnięcie API z reguły stawi czoła walce ponownej (z tzw wariantem podwójnego odstępu w próbach). | Kompletnie wyklucz z tego wyścigu m.in ścieżki w procesach opartych o błędy autoryzacji. W końcu błędne hasło to błędne hasło. |
refetchOnWindowFocus: true | Powrót w zakładce czy nawet kliknięcie po obiekcie do ekranu wyrzuci ponowne przeładowanie zapytań serwera. | Możesz to temperować po wspomnianej wcześniej zaporze czasowej u "StaleTime". |
Co ze zmianami pod dane? Wyprowadzamy Mutacje
Jeden mały wpis a magii znowu mamy pod dostatkiem. Kiedy zapytanie przejdzie test "Zrobiono Pomyślnie" (onSuccess), bufor dla wpisów o tagu komentarzy (dla danego identyfikatora wpisu postId) natychmiast ulega przedawnieniu. Rezultat? Skrypt "TanStacka" wykrzykuje o nowym powiewie informacji od bazy i ładuje na widok zaktualizowaną dyskusję dla każdego użytkownika operującego obecnie pod tym wpisem!
Oczaruj klienta optymistycznymi wczytywaniami (Optimistic updates)
Kto chce czekać po wysłaniu posta by powitać potwierdzenie bycia odebranym od opieszałego pociągu komunikacyjnego dla bazy po np dalekich serwerach Amerykańskich (z tzw strzałów post na logice asynchronicznej)? Zbuduj na ekranach opcję rzucenia powiadomienia a'la Messenger: "Widać że weszło bezbłędnie z mety, czekamy na puste weryfikacje za kulisami w kodzie".
Cały kod zajmuje trochę więcej niż nowe bajery po paczce w postaci haków w najnowszych, bazowych systemach (np useOptimistic od React z lat 19) ale uwierz mi — otrzymujesz na nim wielokrotnie precyzyjniejsze kontrolki kierownicze.
Ciągnące się w nieskończoność skrypty od list (Infinite queries)
Paginacje na "Load More" dla systemów bazodanowych?
Jeśli przy okazji renderujesz bardzo długie listy wyników, połącz to z wirtualizacją, by nie obciążać przeglądarki tysiącami elementów DOM — szczegóły w artykule Wirtualizacja list w React — kiedy TanStack Virtual ratuje FPS.
Co w trawie piszczy o SWR — odświeżenie bez nadwagi
SWR (od sformułowania "stale-while-revalidate") to mniejszy, zwinny kuzyn na rynkach od zaciągania danych, za którego odpowiada z reguły potężna załoga w Vercel. Myśl o nim jak o w pełni podobnej platformie co TanStack z tym by na paczkach ucinać masowe gigabajty (na suche wyniki rzędu ~4 KB na korzyść molocha rzutowanego w rzędzie ~13 KB u starszego z rodziny frameworka dla Query).
Przy bliższym spojrzeniu oba narzędzia mają dużo wspólnego, ale różnią się w kilku istotnych punktach:
Prostota – tu wygrywa SWR. Ma mniejsze API i niższy próg wejścia, więc do prostych projektów i mniejszych aplikacji jest wygodniejszym wyborem.
Bogactwo funkcji – tu wygrywa TanStack Query. Daje znacznie więcej narzędzi do złożonych aplikacji: dynamiczne klucze, rozbudowaną inwalidację cache po mutacjach, optymistyczne aktualizacje i mechanizm anulowania zapytań (query cancellation), który natychmiast przerywa fetch, gdy użytkownik np. cofnie się przed załadowaniem danych.
Typowanie TypeScript – TanStack Query ma dopracowaną integrację z TypeScriptem, z dobrą inferencją typów bez nadmiaru ręcznych adnotacji generycznych.
Reguła wyboru jest prosta: do rozbudowanych aplikacji z wieloma widokami, panelami i mutacjami (dashboardy, SaaS, e-commerce) bierz TanStack Query. Do prostszych projektów — małych paneli, stron marketingowych — wystarczy lżejszy SWR. Oba są solidne i sprawdzone w produkcji.
Jak to gra z Server Components
Złota reguła architektury w App Routerze brzmi: zanim pobierzesz dane na kliencie, zapytaj, czy w ogóle muszą tam trafić. W nowych aplikacjach Next.js większość początkowego pobierania danych powinna dziać się na serwerze, w Server Components:
Do przeglądarki nie trafia wtedy żaden JavaScript fetchujący — serwer pobiera dane i wysyła gotowy HTML. Znikają wodospady zapytań po stronie klienta i koszt wykonania na słabszych urządzeniach. Ten sam model „zero JS domyślnie" opisuję szerzej przy okazji architektury wysp w Astro.
Czego Server Components same nie załatwią? Wszystkiego, co dzieje się interaktywnie po stronie klienta:
- Dynamiczne filtry, paginacja i auto-uzupełnianie — gdy użytkownik klika i odświeża fragment widoku bez przeładowania strony.
- Dane real-time — powiadomienia, czat, statystyki na żywo przez WebSockets albo SSE.
- Mutacje danych — formularze zapisujące zmiany na serwerze, które potem aktualizują widok.
W tych przypadkach wraca TanStack Query (albo SWR) po stronie klienta.
Najlepsze z obu światów — wzorzec prehydracji
Najmocniejsze podejście łączy oba modele: Server Component pobiera dane początkowe, a kliencki TanStack Query przejmuje je jako stan startowy i dalej zarządza odświeżaniem. Użytkownik dostaje gotowy widok natychmiast (dobre dla SEO i pierwszego renderu), a interaktywność i cache działają jak w czystej aplikacji klienckiej:
Dzięki temu zyskujesz zalety obu podejść: serwer renderuje gotowy HTML pod SEO i szybki pierwszy render, a klient dostaje inteligentny cache TanStack Query do odświeżania danych na żywo.
A co z klasycznym useEffect?
Ręczny fetch w useEffect ma dziś już bardzo wąskie zastosowanie. Są w zasadzie trzy sytuacje, w których nadal bywa właściwy:
- Strumienie danych — WebSockets albo SSE, gdzie potrzebujesz utrzymać połączenie i subskrypcję, a nie wykonać pojedyncze pobranie. TanStack Query jest projektowany pod request-response, nie pod ciągły strumień.
- Jednorazowe efekty po załadowaniu — proste akcje, które nie wymagają cache, retry ani inwalidacji.
- Integracje z zewnętrznymi SDK — np. nasłuch zmian autoryzacji albo realtime z Firebase, gdzie podpinasz się pod cykl życia biblioteki.
Poza tymi przypadkami ręczny fetch w useEffect do pobierania danych aplikacji to antywzorzec — generuje wodospady, brak cache, brak retry i problemy z race conditions. Szczegółowo rozbieram to w artykule useEffect to prawie zawsze błąd — kiedy go naprawdę potrzebujesz, a kiedy nie.
Najczęstsze błędy z TanStack Query
W przeglądach kodu klientów te trzy pomyłki wracają najczęściej:
1. Niestabilny klucz zapytania. Jeśli do queryKey wstawisz wartość zmieniającą się przy każdym renderze (np. Date.now()), cache nigdy nie trafi — każde zapytanie ma inny klucz, więc TanStack pobiera dane od nowa za każdym razem.
2. Tworzenie QueryClient wewnątrz komponentu. Jeśli instancja QueryClient powstaje przy każdym renderze (zamiast raz, w stabilnym miejscu), cały cache resetuje się na każdej zmianie drzewa. QueryClient twórz raz — w useState/module — i przekazuj do <QueryClientProvider>.
3. Trzymanie danych serwerowych w Zustand albo Reduxie. Ręczne kopiowanie danych z serwera do globalnego store to praca, którą TanStack Query wykonuje automatycznie — z cache, inwalidacją i odświeżaniem. Globalny store zostaw na stan czysto kliencki (UI, preferencje), a dane serwerowe powierz TanStack Query. Kiedy co wybrać do zarządzania stanem, rozbieram w artykule o Context, Zustand i URL state.
