React js 18 : How to do Data Fetching Techniques with React

Many web applications rely on data from external sources in order to deliver a good user experience. Knowing about the different data fetching techniques in React will enable you make informed decisions on the best use case for your application.

In this article, we will explore various techniques of fetching data in web applications built with React. We will focus on using Hooks introduced in React 16.8 to fetch data from the popular JSON Placeholder API to be displayed in a React application.

Getting Started

Open a terminal and run the command below to create a react application

1# using npm
2npx create-react-app project_name
3
4# using yarn
5yarn create react-app project_name

The command above bootstraps a React application using the create-react-app tool.

Navigate to the project directory

1cd project_name

Start the application

1# using npm
2npm start
3
4# using yarn
5yarn start

Fetch API

The Fetch API is a promise-based Web API that allows you to make network requests in web applications. It is very similar to XMLHttpRequest (XHR) but improves on it by making it easier to make asynchronous requests. It is also better at handling responses.

The fetch API is relatively easy to get started with. Below is an example of using the fetch API to get data:

1fetch('http://jsonplaceholder.typicode.com/posts')
2 .then(response => response.json())
3 .then(data => console.log(data))
4 .catch(error => console.log(error));

In the example above we are fetching data from JSON Placeholder API and outputting it to the console. The fetch takes the location of the resource as an argument and returns a promise as response. We use the response.json() function to convert the response received from the API into JSON. The response can also be converted into other formats as needed by your application. If the promise is rejected due to an error the catch block is executed.

To integrate the fetch API into the React application we created earlier navigate to the src directory and create a directory called components, this folder will hold all the React components. Create a Notes.js file in the components directory, this file is responsible for displaying the data gotten from an external API. Open the file and paste the code below into the file

1import React from 'react';
2const Notes = ({ data }) => {
3 return (
4 <div>
5 <ul>
6 {data && data.map((item, index) => <li key={index}>{item.title}</li>)}
7 </ul>
8 </div>
9 );
10};
11export default Notes;

Create another file called FetchDemo.js and paste the code below

1import React, { useState, useEffect } from 'react';
2import Notes from './Notes';
3const FetchDemo = () => {
4 const [notes, setNotes] = useState([]);
5 const [isLoading, setIsLoading] = useState(false);
6 const [isError, setIsError] = useState(false);
7 const fetchData = () => {
8 fetch('http://jsonplaceholder.typicode.com/posts')
9 .then((response) => response.json())
10 .then((data) => {
11 setIsLoading(false);
12 setNotes(data);
13 })
14 .catch((error) => {
15 setIsLoading(false);
16 setIsError(true);
17 console.log(error);
18 });
19 };
20 useEffect(() => {
21 fetchData();
22 }, []);
23 if (isLoading) {
24 return <div>Loading...</div>;
25 }
26 return (
27 <div>
28 <h1>Fetch Example</h1>
29 {notes && <Notes data={notes} />}
30 {isError && <div>Error fetching data.</div>}
31 </div>
32 );
33};
34export default FetchDemo;

In this component, we’ve created a fetchData function that is responsible for getting a list of notes from JSON Placeholder. The response gotten from the data is then stored in the notes state which is passed as props {notes && <Notes data={notes} />} to the Notes component we created earlier for displaying the notes. If there is an error encountered when fetching the data we modify the isError state and display an error message to the user {isError && <div>Error fetching data.</div>}. We also have a state isLoading which we use to display a loading message to the user while we wait for the response from the network request made.

The fetchData function is called in the useEffect hook which runs once when the component is mounted (for more on useEffects check here).

Go to the App.js file and replace the existing code with the code below

1import React from 'react';
2import FetchDemo from './components/FetchDemo';
3function App() {
4 return (
5 <div className="App">
6 <h1>Posts</h1>
7 <FetchDemo />
8 </div>
9 );
10}
11export default App;

What we’ve done here is to import the FetchDemo component we created and render it in the App component. Save the file and see the results in your browser, the results should be similar to the screenshot below:

null

Open Source Session Replay

OpenReplay is an open-source, session replay suite that lets you see what users do on your web app, helping you troubleshoot issues faster. OpenReplay is self-hosted for full control over your data.

replayer.png

Start enjoying your debugging experience - start using OpenReplay for free.

Axios

Axios is a promise-based lightweight HTTP client for the browser and Node.js. It is similar to the Fetch API as it is also used in making network requests. Unlike fetch, it transforms all responses to JSON.

Below is an example of using the fetch API to get data

1axios.get('http://jsonplaceholder.typicode.com/posts')
2 .then(response => console.log(response))
3 .catch(error => console.log(error));

This example is very similar to fetch as Axios is also promise-based. The major difference as mentioned earlier is we don’t have to convert the responses to JSON as Axios does that automatically. Some other features of Axios include:

  • Intercept request and response
  • Transform request and response data
  • Cancel requests
  • Client-side support for protecting against XSRF

To integrate Axios into a React application open a terminal in the applications root directory and run the code below

1# using npm
2npm install axios
3
4# using yarn
5yarn add axios

The above commands installs Axios into a React application.

Create a file called AxiosDemo.js in the components directory and paste the code below

1import React, { useState, useEffect } from 'react';
2import axios from 'axios';
3import Notes from './Notes';
4const AxiosDemo = () => {
5 const [notes, setNotes] = useState([]);
6 const [isLoading, setIsLoading] = useState(false);
7 const [isError, setIsError] = useState(false);
8 const fetchData = () => {
9 axios
10 .get('http://jsonplaceholder.typicode.com/posts')
11 .then((response) => {
12 setIsLoading(false);
13 setNotes(response.data);
14 })
15 .catch((error) => {
16 setIsLoading(false);
17 setIsError(true);
18 console.log(error);
19 });
20 };
21 useEffect(() => {
22 fetchData();
23 }, []);
24 if (isLoading) {
25 return <div>Loading...</div>;
26 }
27 return (
28 <div>
29 <h1>Using Axios</h1>
30 {notes && <Notes data={notes} />}
31 {isError && <div>Error fetching data.</div>}
32 </div>
33 );
34};
35export default AxiosDemo;

The code above shows how to fetch data using Axios. The first step was importing the Axios library we installed import axios from 'axios' the Axios library is then utilized in the fetchData function where we make a request to get data from the JSON Placeholder library. We store the response gotten in the notes state and if we encounter any error, we render an appropriate error message to the user.

Go to the App.js file and replace the existing code with the code below

1import React from 'react';
2import FetchDemo from './components/FetchDemo';
3import AxiosDemo from './components/AxiosDemo';
4function App() {
5 return (
6 <div className="App">
7 <AxiosDemo />
8 </div>
9 );
10}
11export default App;

In the App component we’ve rendered the AxiosDemo component. Save and open the application in a browser

null

Fetching data with Async/Await

“async/await” makes asynchronous programming easier in JavaScript. It provides a way of writing promises cleanly and concisely and takes away all .then() blocks. There are two parts to “async/await”:

  • async: The async is a keyword placed before a function declaration which makes the function return a promise. For example: async function hello() { return “hello”; } hello().then(console.log) //hello
  • await: The await keyword is used in an async function and makes the function wait for a promise to be resolved. let value = await promise;

“async/await” is less of a data fetching technique and more of a better way of using the existing data techniques mentioned earlier. We can refactor the existing Axios network call using promises in the previous section from the code below

1axios
2 .get('http://jsonplaceholder.typicode.com/posts')
3 .then((response) => {
4 setIsLoading(false);
5 setNotes(response.data);
6 })
7 .catch((error) => {
8 setIsLoading(false);
9 setIsError(true);
10 console.log(error);
11 });

to:

1try {
2 const response = await axios.get(
3 'http://jsonplaceholder.typicode.com/posts'
4 );
5 setIsLoading(false);
6 setNotes(response.data);
7} catch (error) {
8 setIsLoading(false);
9 setIsError(true);
10 console.log(error);
11}

This makes the code less complex and easier to read.

Custom Data Fetching Hooks

According to the React docs:

A custom Hook is a JavaScript function whose name starts with ”use” and that may call other Hooks

Custom hooks allow you to extract component logic into reusable functions that can then be used in different components. For example, if we have to fetch data across different components instead of having the data fetching logic in each of these components we can abstract that logic into a custom hook which can then be reused across the different components.

We will be creating a useAxios hook that takes away the data fetching logic from the AxiosDemo component we created earlier. Navigate to the src directory and create a hooks directory. Now go to the hooks directory and create a file called useAxios.js. Open the file and paste the code below:

1import { useState, useEffect } from 'react';
2import axios from 'axios';
3
4const useAxios = (url) => {
5 const [data, setData] = useState([]);
6 const [isLoading, setIsLoading] = useState(false);
7 const [isError, setIsError] = useState(false);
8 const fetchData = async () => {
9 try {
10 const response = await axios.get(url);
11 setIsLoading(false);
12 setData(response.data);
13 } catch (error) {
14 setIsLoading(false);
15 setIsError(true);
16 console.log(error);
17 }
18 };
19 useEffect(() => {
20 fetchData();
21 }, []);
22 return { isLoading, isError, data };
23};
24export default useAxios;

In the code above we’ve extracted the states and data fetching logic needed from the AxiosDemo component, we’ve also added a parameter for url. What this means is we can use this hook in any component that needs to fetch data from anywhere not necessarily JSON Placeholder and return a response back to the component that needs the data.

Now let us refactor our AxiosDemo component to use the useAxios hook. Open AxiosDemo.js file and replace the existing code with the code below:

1import React from 'react';
2import useAxios from '../hooks/useAxios';
3import Notes from './Notes';
4
5const AxiosDemo = () => {
6 const { isLoading, isError, data: notes } = useAxios(
7 'http://jsonplaceholder.typicode.com/posts'
8 );
9 if (isLoading) {
10 return <div>Loading...</div>;
11 }
12 return (
13 <div>
14 <h1>Using Axios</h1>
15 {notes && <Notes data={notes} />}
16 {isError && <div>Error fetching data.</div>}
17 </div>
18 );
19};
20
21export default AxiosDemo;

We’ve imported the useAxios hook into the AxiosDemo component. We use the hook by passing the url argument:

1const { isLoading, isError, data: notes } = useAxios("http://jsonplaceholder.typicode.com/posts");

Once the data is fetched the hook returns an object which we have destructured to be used in the AxiosDemo component.

Concurrent Mode and Suspense

Concurrent mode and Suspense are experimental features introduced in React 16.6 to handle data fetching in React applications. To get started with concurrent mode, open a terminal in the root of your application and run the code below:

1# using npm
2npm install react@experimental react-dom@experimental
3
4# using yarn
5yarn add react@experimental react-dom@experimental

Go to index.js and replace the existing code with the code below:

1import ReactDOM from 'react-dom';
2
3// If you previously had:
4//
5// ReactDOM.render(<App />, document.getElementById('root'));
6//
7// You can opt into Concurrent Mode by writing:
8
9ReactDOM.unstable_createRoot(
10 document.getElementById('root')
11).render(<App />);

Concurrent mode is a set of new features that helps React apps stay responsive while waiting for the result of an operation, this includes Suspense which is a component that lets you specify a loading state while you wait for some code action to be done executing.

1// show a loading message while waiting for notes to be loaded
2const notes = fetchData()
3<Suspense fallback={<div>Loading...</div>}>
4 <Notes data={notes} />
5</Suspense>

In the example above we are displaying a loading message while waiting for the data from our API to be loaded. You might wonder how this differs from using a loading state as we’ve used in previous sections.

1if (isLoading) {
2 return <div>Loading...</div>;
3}

As an application gets bigger we might have to depend on different loading states when each component has to make a different asynchronous request which makes our code get messy. Since Suspense knows when our data is fetched it saves us a lot of time in boilerplate code and having to listen for loading state changes in our application.

To be clear Concurrent mode and Suspense is still experimental and shouldn’t be used in production-ready applications. It is also good to note that Suspense is not a data fetching mechanism but rather a way to delay the rendering of components while you wait for unavailable data. Dan Abramov gave a great demo on how Suspense works check it out here.

Conclusion

In this article we’ve looked at different data fetching techniques for React applications, while the approaches are similar it is left for you to choose the best approach to suit the use case of your application.