Fetching with useEffect
We've already learned about one hook, useState
. Time for another one! In this lesson, we'll learn how to use the useEffect
hook to send API fetch requests.
Table of Contents
Terms
Side effect — Anything that happens outside of React such sending a
fetch
request, starting an animation, or setting up a server connection.Side effects can be triggered by user events like submitting a form or clicking a button.
useEffect
– A react hook for executing "side effects" caused by a component rendering, not a particular event.Hooks — Functions that provide a wide variety of features for React components. They all begin with
use()
.
Dependency Array — The array of values provided to
useEffect
that React will watch for changes. If changes occur in the dependency array, the effect will run again.Conditional Rendering — Rendering different JSX depending on the current state. This can be useful when fetching to show either the fetched data or an error message if the fetch failed.
Fetching with event handlers
For a refresher on how to fetch, look at the
src/utils/fetchData.js
helper function. It returns an array with two values[data, error]
(a "tuple").
In React, sending a fetch request is referred to as an "effect" or "side effect" since it happens outside of the normal scope of what React handles.
Side effects are often executed in event handlers.
Check out the 1-joke-fetch-on-click
React project. In our application, we can render utilize a random joke API to send a fetch request in response to a button click:
This example demonstrates a few important concepts:
When fetching, the fetched data should be stored in state (
joke
)We should also make a piece of state to store an error if one is returned (
error
)We can use conditional rendering to render an error message if there was one.
Challenge 1: Make a Dog API app
Let's create an app that fetches from the dog API and shows a random dog picture whenever the user clicks on a button.
The dog API https://dog.ceo/api/breeds/image/random returns an object like this:
Instructions:
Create the app
Then, copy the the
src/utils
folder from the1-joke-fetch-on-click
folder into your ownsrc/
folder.Use the code in the
1-joke-fetch-on-click/src
folder to guide you to creating this appThe
App
should have adogPicture
and anerror
state. Visit the API https://dog.ceo/api/breeds/image/random to get an example object that you can use as the starting state fordogPicture
. Something like this:
Replace the
App
contents with your own app that has a<button>
and an<img>
. Theimg
should render thedogPicture.message
.When the user clicks on the button, it should send a fetch to the dogAPI and update either the
dogPicture
orerror
state depending on the returned tupleAdd a conditional render to show the
error.message
if there is an error.
useEffect
There are two ways to perform a side effect like fetching:
In response to user events
In response to the component rendering ("reacting to the component rendering")
In our current joke API app, we only send a fetch in response to the user clicking on the button. But what if we want to show a joke when the page first renders?
We can accomplish this with the hook useEffect
— a react hook for executing "side effects" caused by a component rendering, not a particular event.
Q: How do we know that this is a hook?
useEffect Syntax
useEffect
takes in two arguments:
A callback function
[optional] A "dependency array"
It should be invoked at the top of the component, next to the other hooks used by the component (often below useState
)
Notice that this callback creates a async doFetch
function that fetches, and sets the joke
or the error
state depending on what is returned.
The Effect callback
Why do we need to define doFetch
and then invoke it? Why not just make the callback itself async.
Unfortunately, we can't make the callback async — we get an error
So, inside of the callback, we make an async
function that does the fetch and then invoke it immediately.
The Dependency Array
useEffect(effect, dependencyArray)
needs to accept an effect
callback but the second argument dependencyArray
is optional. There are three ways that we can provide this value:
If the array is omitted, the effect is executed on EVERY render of the component.
If the array is empty, the effect is only executed on the first render of the component.
If the dependency array is provided, the effect will be only re-run on future renders if the values in the array change between renders.
Challenge 2: Fetch On Render
Add to your dog API app by having it render a dog image on the first render (and only on that first render!)
Fetching With a Form On Change
A cool way to fetch is using a form whenever the text input changes:
Quiz
When should you
fetch
usinguseEffect
vs. an event handler?
Last updated