data:image/s3,"s3://crabby-images/0856d/0856d2888a6e3d21abf64526c330a02cfdeb1565" alt=""
data:image/s3,"s3://crabby-images/4f136/4f13638c58b2550d14476d53103521638ab40d09" alt="AbortController when doing Fetch API"
AbortController when doing Fetch API
If you're accustomed to using the Fetch API in React (or Preact), you're likely familiar with the following code structure.
const Home = () => {
const [users, setUsers] = useState([]);
const [error, setError] = useState("");
useEffect(() => {
const endpoint = "https://jsonplaceholder.typicode.com/users";
fetch(endpoint).then(response => {
return response.json();
}).then(newUsers => {
setUsers(newUsers);
}).catch(({message}) => {
setError(message);
});
}, []);
return ({users.map(({username}, key) => ({ username }))});
};
However, what occurs when your network connection suddenly slows down? Perhaps the network response is delayed, prompting you to navigate to another page.
Coincidentally, at that very moment, on a separate page, you initiate another request using the same pattern. Consequently, you find yourself with two simultaneous requests vying for the limited network bandwidth.
Naturally, you begin to question: is my network experiencing issues? To investigate, you attempt to access another page with the same pattern, further burdening the server with three concurrent connections over a sluggish network.
Fortunately, this issue can be easily addressed by employing an AbortController
.
const Home = () => {
const [users, setUsers] = useState([]);
const [error, setError] = useState("");
useEffect(() => {
const endpoint = "https://jsonplaceholder.typicode.com/users";
// Instanciation of our controller
const controller = new AbortController();
// Attaching the signal to the request
fetch(endpoint, {signal: controller.signal}).then(response => {
return response.json();
}).then(newUsers => {
setUsers(newUsers);
}).catch(({message}) => {
setError(message);
});
// Canceling the request when the component is destroyed
return () => controller.abort();
}, []);
return ({users.map(({username}, key) => ({ username }))});
};
I've inserted a comment above the new additions. These three lines suffice to prevent background requests from overwhelming the network unnecessarily.
Now, whenever a user navigates to another page, the cleanup function triggers, halting the request via the abort controller. This action conserves valuable bandwidth for subsequent requests, hopefully ensuring success.
Interestingly, the same principle applies to Vue.js, where you can invoke the controller within the destroyed
lifecycle method in Vue 2 and onBeforeUnmount
in Vue 3.
This is sample code which using Vue 3:
import { ref, onBeforeUnmount } from 'vue';
const endpoint = "https://jsonplaceholder.typicode.com/users";
const controller = new AbortController();
const error = ref("");
const users = ref([]);
fetch(endpoint, { signal: controller.signal })
.then(response => response.json())
.then(newUsers => users.value = newUsers)
.catch(({ message }) => error.value = message);
onBeforeUnmount(() => {
controller.abort();
});
Thanks for reading!
Related Blogs
data:image/s3,"s3://crabby-images/0856d/0856d2888a6e3d21abf64526c330a02cfdeb1565" alt=""
data:image/s3,"s3://crabby-images/5ec40/5ec40b65249cb7588858fad175fb82524915fef1" alt=""
data:image/s3,"s3://crabby-images/5d9d3/5d9d38a499f97214b567ae87f2e3b5c5c5bf3e75" alt=""
data:image/s3,"s3://crabby-images/7b1d8/7b1d88bd685bf8b989c93584fa98d53eda4dc545" alt=""
data:image/s3,"s3://crabby-images/f27dc/f27dc6a1e670ec35d511e3ad6347ccee57061507" alt=""
data:image/s3,"s3://crabby-images/65312/653121ecacf60f7a3985111cc6455e9a0e2aca56" alt=""
data:image/s3,"s3://crabby-images/88873/888733ba145445c5ce3b6d00ec6e0fe0155f8cde" alt=""
data:image/s3,"s3://crabby-images/c0c16/c0c16f401ad66d71e0191c4720734497aefe6b06" alt=""
data:image/s3,"s3://crabby-images/d280d/d280db064f284647032f51394dcb5a376be52e9a" alt=""
data:image/s3,"s3://crabby-images/706b0/706b0cf737814ae6f94c518e7e4ae0f083d9c02e" alt=""
data:image/s3,"s3://crabby-images/defd9/defd99772b914501db4f7771363fdfde07a507b2" alt=""
data:image/s3,"s3://crabby-images/917e1/917e1755c102dc28cb3e763cd9d8dce74ed9b554" alt=""
data:image/s3,"s3://crabby-images/f76a0/f76a0add21c8d12d98b198ab674cb0001d344392" alt=""
data:image/s3,"s3://crabby-images/dea89/dea89e5ca091c3abbba60b6cd6fe7b461f054e34" alt=""
data:image/s3,"s3://crabby-images/6215d/6215d836d45359f0f4f60c787d63873243115929" alt=""
data:image/s3,"s3://crabby-images/52d05/52d057e8d8c4176e832d565ff57f48727638c216" alt=""
data:image/s3,"s3://crabby-images/fa4c3/fa4c3bc5b59c1417b4b2cff4afabbd91cd5c0349" alt=""