Skip to content

Learns about how to effectively use promises for dealing with asynchronous code

License

Notifications You must be signed in to change notification settings

cnguyen0/promises-challenge

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

17 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

promises-challenge

What's a promise?

As you already know, when doing web programming you'll often make calls to a foreign resource that takes some time load. If the browser just stopped and waited until that thing loaded, the web would seriously be awful to use. So instead, we program asynchronously, which allows us to continue doing other things while we wait for our programs to load.

What's a 'promise' in the world of JavaScript? It's a nicer way of dealing with callbacks, and handling asychronous code based on a common specification. The exact specification that all Promise implentations follow is known as A+. Now that defines what all Promise specifications have to follow, but many implementations add extra functions that are very useful.

We'll be using jQuery's promises, because you're already familiar with it, it's almost always available. Take a look at their deferred object API. Their deferred objects are basically promises, but have slightly more functionality available to them.

Promise basics

Promises are great, because they're so easy to use!

var p = Promise(); // Not real, just pretend it's a promise

p.then(success, fail);

p is a promise object, and one of the methods available is then. It takes two functions, a success function and a failure function. So if the promise failed (like making a web request that returned a 404) it will do whatever failure does. But if it succeeds, it will do whatever success does. This is a consise way to specify behaviour asynchronously. Note that failure is optional, and you could instead just pass the success calback.

We can also specify this same behavior with done() and failure(). Take a look at the following:

p.done(success).fail(fail);

It's the exact same as the above! However, done() and fail() are useful because you can specify multiple callbacks for each situation, not just one:

p.done(success1, success2, successN);
  .fail(failure1, failure2, failureN)

In addition, there's always(). It, as you might expect, will always perform whatever is passed into it, whether the promise succeeds or fails!

p.then(succ, fail).always(doThisNoMatterWhat);

So the promise above would either call succ() or fail(), but it will call doThisNoMatterWhat() every time.

Multiple Promises

The above goes over how to deal with a single promise, but the really cool part of them is that it's easy to specify complex behavior for groups of actions by chaining promises together.

Serial actions

Let's say you want do several things one by one. Like making 3 web requests, that all depend on each other.

var req1 = function() {
  return $.getJSON(something);
}

var req2 = function() {
  return $.getJSON(somethingElse);
}

var req3 = function() {
  return $.getJSON(yetAnotherThing);
}

req1()
  .then(req2)
  .then(req3)
  .done(function() {console.log('Everthing is awesome!')})
  .fail(function() {console.log('Oh Noes, there was an error')});

What the above does is construct a promise that execute each request one by one, and if any of them fail along the way, then our console will inform us there was an error. The reason this works is because req object returns a new promise object, which we can then call more promise methods on!

var serialPromise = req1.then(req2).then(req3);
serialPromise
  .done(successFunction)
  .fail(failFunction);

In this case, we aren't doing anything that would actually require each request to be done one by one, but there are certainly situations where you will have things should happen one by one, and this is how you set it up.

Parallel requests

So the above would actually be bad, because we're wasting the users time with each request. If the requests don't depend on each other, we should perform them at the same time and then do something one all of them are finished. Promises make this incredibly simple of course!

// Assume the req objects from above are defined the same way as jQuery AJAX calls
var parallelPromise = $.when(req1, req2, req3)

parallelPromise.
  .done(function( response1, response2, response3 ){
  	// Do something with all of the independent responses
  })
  .fail(failureFunction);

Here we use jQuery's handle when() function for doing paralell requests. When takes an arbitrary number of promises, and returns a promise. The promise takes a callback function that then has values bound for each result of the promises! What's cool is each request will occur at the same time, and we only start doing things once all of them are finished. Often in web applications, your greatest bottleneck is the network, so efficiently utilizing the network will have huge perfomance impact.

Note that when() is specific to jQuery! Not all promise libraries have handy methods for handling parallel promises, in fact some libraries do a much better job (by allowing you to deal with promises in arrays). Unfortunately the core specification doesn't have such a useful function.

About

Learns about how to effectively use promises for dealing with asynchronous code

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • JavaScript 52.2%
  • HTML 39.6%
  • CSS 8.2%