This repo was created using create-react-app
Clone this repo and the promises-api repo down.
For both:
npm install
npm start
Understanding of:
- Why promises over callbacks
- How to implement a promise
- How to use Promise.all()
Brief Understanding of:
- single-threaded JS
- asynchronous JS
- non-blocking
- call stack
- task queue
- event loop
- heap
Javascript is a single-threaded, non-blocking, asynchronous, concurrent language
. That's a mouth full... Let's break things down into sections that are a bit more manageable.
As we know javascript has a single-threaded call stack
that has stack frames. As our code is run, each stack frame is pushed and popped from the top of the call stack, the top being the current code being executed. Think of a can of Pringles... the first in is the last out (FILO).
Check out this awesome video if you want a deeper dive into the call stack
, task queue
, event loop
, web APIs
and how they all work together VIDEO
So consider the javascript:
function foo(b) {
var a = 10;
return a + b + 11;
}
function bar(x) {
var y = 3;
return foo(x * y);
}
console.log(bar(7));
So what we have is a function foo(b)
that returns 10 + b + 11
.
Then there is bar(x)
that returns foo(x * y)
.
Our console.log ends up calling bar(7)
Lets watch the call stack:
As we see the stack starts by pushing console.log(bar(7))
because it was the first executed code we have. Which then calls bar(7)
to execute foo(x * y)
to execute the inner operation x * y
. So far we have just been pushing things to the stack. Until after x * y
has finished executing then it gets popped off!
Now we continue on with foo(x * y)
which pushes a + b + 11
to the top of the stack to immediately get popped off leaving a + b
to be pushed to the top. Then finishes off by popping all the rest of the stack. This will finish by returning 42
to the console.
Go try it out here. Did anything happen you didn't expect? Talk to a neighbor about what you've learned after playing around with the stack.
So! From the image above we can start to understand these these concepts:
Single threaded:
Threads are basic units of CPU utilization.Asynchronous:
JavaScript call stack carries out tasks to completion instead of task switching and the same holds for events.Non-blocking:
Blocking occurs when the application state is suspended as a thread runs.
A very interesting property of the event loop model is that JavaScript, unlike a lot of other languages, never blocks. Handling I/O is typically performed via events and callbacks, so when the application is waiting for a network request to return, it can still process other things like user input.*
Heres a great example:
Example --> When Big Loop handler runs the browser appears frozen. We know JavaScript’s call stack is synchronous soBig Loop executes on the call stack until completion. It’s also non-blocking where Do Stuff clicks are still received even if they didn’t execute immediately.
Let's continue on with asynchronous. With non-blocking code, we can have a user click a button and continue on with I/O without making them wait.
How does that work? Well for a long time the web has used callbacks
to help solve this issue, allowing code to be executed once its finished. For example, the browser
has web api's that take callbacks as an argument, like: XHR
, SetTimeout
, DOM
events. These things have their own callback queues
/task queues
which run an event loop
.
Here's an example:
try it yourself
When your code first executes, it loads events and saves them in heap
, but for now lets reference it by the web api box. So the call stack starts running through our code and setTimeout(function timeout{...})
(line 9) gets put on the call stack. Which sends it over to the web api box for storage until it's ready to execute the callback after 5 seconds! Once the wait is done, it will send the callback to the callback queue / task queue
while the event loop
waits for a good time to throw it on the stack / till the stack is cleared. ( event loop is the orange loop in this example )
Now we also saw that the DOM event listener
, when put on the call stack(line 1) is pushed to the web apis storage and waits/listens for its time to be called. You'll notice that once I trigger the event by clicking the 'Click me!' button, that callback is pushed to the callback queue / task queue
. Once the stack has cleared the event loop
will trigger the next callback.
So what does this mean? Yes, your JS code is running on a single call stack (single-threaded). But under your JS, the browser code is running multiple threads to manage async actions. When an async action is done doing its thing, the callback gets pushed on the task queue. The event loop is constantly checking to see if the call stack is empty, and when it is, the task queue grabs the first thing in the queue and puts it on the call stack. And this is where the function is executed.
Hope that review was helpful!
What we have so far is an api that serves up a collection of members here. We also have our client side code located on this repo.
So the API given to us doesn't automatically give us all the info needed to display the staff members. There is a second nested endpoint we will need to fetch..
The endpoints given to us are:
- http://localhost:3001/api/frontend-staff - this returns an object with a key of bio and the value is an array of objects that contain the name of each staff member and another endpoint to grab some additional info (bio & image).
- http://localhost:3001//api/bio/:id - this is the endpoint given from each obj inside the array from the endpoint
frontend-staff
- make initial fetch
- map over the array of staff members
- I need to fetch each endpoint
- this map is going to return a promise
- Promise.all(promises)
- returns a single promise
The first thing we need ask ourselves is where do we want to fetch our data from. This article does a really solid job of answering that question. Looks like componentDidMount() is our best option. With componentDidMount(), We can be assured that our component has mounted on the DOM and we can also setState there once we get our data back.
If you're not feeling totally comfortable with fetch
yet, I suggest taking a five minutes to review the docs.
Fetch returns a promise, which will either resolve
or reject
depending on the status of the promise. You might want to take a look at when fetch actually catches errors here. The api can actually be set up in a way that can help fix this, but this is a major reason why some people dislike fetch
.
So, since fetch
returns a promise, it makes sense that you can chain .then()
or .catch()
to it.
componentDidMount() {
const url = 'http://localhost:3001/api/frontend-staff'
fetch(url)
.then(response => response.json())
.then(data => this.fetchBios(data.bios)) // we'll write this function that will fetch all of the nested endpoints shortly
.then(staff => this.setState({ staff }))
.catch(error => console.log(error))
}
When we make a request and the response comes back as JSON, the first thing we need to do is parse the JSON. Notice also that we are chaining another .then() to the response.json(). We can do this because .json() returns a promise that resolves with the result of parsing the body text as JSON.
Now that we have our initial fetch set up, we need to iterate over the array to fetch the other information we need. Yes, we could make 10 different fetch calls, but all our data is going to come back at different times. That's not going to make for a great UI experience. How can we fetch the additional data, but wait for all the data to be returned before we do anything with it? Enter Promise.all()
So Promise.all() takes an array of promises and returns a single promise that will either resolve
when every promise has resolved or reject
with the reason of the first value in the array that reject. If the promise array resolves completely, the resulting values will be an array of values whose results are ordered by the order of the promises in the original array - regardless of which promises resolve first.
How can we use this to our advantage? So when we make our request to 'api/frontend-staff' we receive an array
of staff members containing info to make more fetch calls
.
fetch('http://localhost:3001/api/frontend-staff')
.then(response => response.json())
// this returns =>
{ bio: [
{ name: "Louisa Barrett",
info: "http://localhost:3001/api/bio/1"
},
{...},
{...}
]
}
So we're probably going to have to iterate over this array to make a fetch call for all the bios and images.
fetchBios = (staff) => {
const promises = staff.map(staffMember => {
return fetch(staffMember.info)
.then(response => response.json())
.then(data => ({...data, name: staffMember.name}))
.catch(error => console.log(error))
})
return Promise.all(promises)
}
- MDN docs
- Loupe, by Philip Roberts, was used for examples.
- DAN MARTENSEN wrote this article that I referenced his code pen from.