Get data elegantly using React hooks
Write it up front
- React hooks Use react hooks.
- This article mainly writes about how to use
state
和effect
Hooks to gracefully retrieve list data.- How to customize a hook to fetch data?
- Of course you need to check out the new features of React Hooks first
- Github.com/reactjs/rfc…
- Reactjs.org/docs/hooks-…
Use hook to get data
- How do react gracefully fetch data
- Now let’s see how to use hook to get
1. Use of useState
import React, { useState } from 'react';
function App() {
const [data, setData] = useState({ hits: []});return (
<ul>
{data.hits.map(item => (
<li key={item.objectID}>
<a href={item.url}>{item.title}</a>
</li>
))}
</ul>
);
}
export default App;
Copy the code
Automatically call back to the userState method to store data in state.
2. Use of Axios (useEffect)
import React, { useState, useEffect } from 'react'; import axios from 'axios'; function App() { const [data, setData] = useState({ hits: [] }); useEffect(async () => { const result = await axios( 'http://hn.algolia.com/api/v1/search?query=redux', ); // When using useEffect, we actively set 'state' and store 'setData' setData(result.data); }); return ( <ul> {data.hits.map(item => ( <li key={item.objectID}> <a href={item.url}>{item.title}</a> </li> ))} </ul> ); } export default App;Copy the code
When you run the above program, you will find a bug, why is this? Because when we store the data in state after we get it, our component will update and Effect will run again. And then we’re going to get data again.
-
Warning: useEffect function must return a cleanup function or nothing. Promises and useEffect(async () => …) are not supported, but you can call an async function inside an effect..
-
How to avoid the above problems?
-
The goal is to get the data only when the component is loaded
import React, { useState, useEffect } from 'react';
import axios from 'axios';
function App() {
const [data, setData] = useState({ hits: []});// Separate fetchData
const fetchData = async() = > {const result = await axios(
'http://hn.algolia.com/api/v1/search?query=redux',); setData(result.data); };// The reason for splitting fetchData separately is that async in effect is not recommended in the waring section above
useEffect(() = >{ fetchData(); } []);// The second argument here is: hooks to watch for changes in the value
// Add [] here, go back to see if effect has changed when our component updates, add empty [] here to prevent hooks from running again.
return (
<ul>
{data.hits.map(item => (
<li key={item.objectID}>
<a href={item.url}>{item.title}</a>
</li>
))}
</ul>
);
}
export default App;
Copy the code
3. How to use the Form?
- When you use a Form
function App() {...// In order to prevent the default behavior of the form, we recommend splitting the doGet method to write separately. Instead of writing it directly in the formIn the onchange method ofconst doGet = event= > {
setUrl(`http://hn.algolia.com/api/v1/search?query=${query}`);
event.preventDefault();
};
return (
<Fragment>
<form onSubmit={doGet}>
<input
type="text"
value={query}
onChange={event= > setQuery(event.target.value)}
/>
<button type="submit">Search</button>
</form>
{isError && <div>Something went wrong ...</div>}...</Fragment>
);
}
Copy the code
4. How to customize a data acquisition hook?
- Of course, we can customize a hook according to our own situation to obtain our data, which needs to deal with loading, error, data source, etc.
// Make a news data acquisition API
const useHackerNewsApi = () = > {
const [data, setData] = useState({ hits: []});const [url, setUrl] = useState(
'http://hn.algolia.com/api/v1/search?query=redux',);const [isLoading, setIsLoading] = useState(false);
const [isError, setIsError] = useState(false);
const fetchData = async () => {
setIsError(false);
setIsLoading(true);
try {
const result = await axios(url);
setData(result.data);
} catch (error) {
setIsError(true);
}
setIsLoading(false);
};
useEffect(() = > {
fetchData();
}, [url]);
const doGet = event= > {
setUrl(`http://hn.algolia.com/api/v1/search?query=${query}`);
event.preventDefault();
};
return { data, isLoading, isError, doGet };
}
Copy the code
- How to use
// Just call it
function App() {
const [query, setQuery] = useState('redux');
const { data, isLoading, isError, doGet } = useHackerNewsApi();
return (
<Fragment>.</Fragment>
);
}
Copy the code
Further customizationAPI
的 url
const useHackerNewsApi = () = > {
...
useEffect(
...
);
const doGet = (event, url) = > {
setUrl(url);
event.preventDefault();
};
return { data, isLoading, isError, doGet };
};
function App() {
const [query, setQuery] = useState('redux');
const { data, isLoading, isError, doGet } = useHackerNewsApi();
return (
<Fragment>
<form
onSubmit={event= >
doGet(
event,
`http://hn.algolia.com/api/v1/search?query=${query}`,
)
}
>
<input
type="text"
value={query}
onChange={event= > setQuery(event.target.value)}
/>
<button type="submit">Search</button>
</form>.</Fragment>
);
}
Copy the code
customstate
Initialization of data
import React, { Fragment, useState, useEffect } from 'react';
import axios from 'axios';
// Customize initialUrl and initialData
const useDataApi = (initialUrl, initialData) = > {
const [data, setData] = useState(initialData);
const [url, setUrl] = useState(initialUrl);
const [isLoading, setIsLoading] = useState(false);
const [isError, setIsError] = useState(false);
const fetchData = async () => {
setIsError(false);
setIsLoading(true);
try {
const result = await axios(url);
setData(result.data);
} catch (error) {
setIsError(true);
}
setIsLoading(false);
};
useEffect(() = > {
fetchData();
}, [url]);
const doGet = (event, url) = > {
setUrl(url);
event.preventDefault();
};
return { data, isLoading, isError, doGet };
};
function App() {
const [query, setQuery] = useState('redux');
const { data, isLoading, isError, doGet } = useDataApi(
'http://hn.algolia.com/api/v1/search?query=redux',
{ hits: []});return (
<Fragment>
<form
onSubmit={event= >
doGet(
event,
`http://hn.algolia.com/api/v1/search?query=${query}`,
)
}
>
<input
type="text"
value={query}
onChange={event= > setQuery(event.target.value)}
/>
<button type="submit">Search</button>
</form>
{isError && <div>Something went wrong ...</div>}
{isLoading ? (
<div>Loading ...</div>
) : (
<ul>
{data.hits.map(item => (
<li key={item.objectID}>
<a href={item.url}>{item.title}</a>
</li>
))}
</ul>
)}
</Fragment>
);
}
export default App;
Copy the code
Now we can gracefully use hooks to retrieve our data
reference
- Juejin. Cn/post / 684490…
- Reactjs.org/docs/hooks-…
- www.robinwieruch.de/react-fetch…