-
-
Notifications
You must be signed in to change notification settings - Fork 83
Comparison to Promises
As mentioned in the readme, Futures are conceptually very similar to Promises. This wiki page aims to document the differences between the two control structures.
Futures can be thought of as the "algebraic" counterpart to Promises. When the Promise/A+ spec was drafted, an issue was raised to make Promises conform to the "monadic interface": A collection of container-related behaviors defined by the mathematical field of category theory. The people behind the Promise specification wanted nothing of it:
Yeah this is really not happening. It totally ignores reality in favor of typed-language fantasy land -- comment
This is how the Fantasy Land algebraic specification was born and incidentally, where it got its name. Fantasy Land defines the interface, function signatures, and laws that a structure must conform to in order to be monadic.
Fluture provides a structure with equal capabilities to Promises but with a monadic API. It can therefore be thought of as Promise's more algebraic cousin.
new Promise((res) => setTimeout(200, res, 'Hello'))
.then(x => `${x} world!`)
.then(x => Promise.fromNode(done => fs.writeFile('hello.txt', x, done)))
.then(console.log, console.error);
//vs
Future((rej, res) => setTimeout(200, res, 'Hello'))
.map(x => `${x} world!`)
.chain(x => Future.node(done => fs.writeFile('hello.txt', x, done)))
.fork(console.error, console.log);
On the surface Futures are just like Promises, but with the different behaviors of the .then
method extracted into three distinct functions, each with a single responsibility:
-
map
: Transforms the success value inside the Future. This happens in Promise if you return anything that doesn't look like a Promise fromf
in.then(f)
. -
chain
: Absorb the state of another Future into the main Future (flat map). This happens in Promise if you return something that looks like a Promise fromf
in.then(f)
. -
fork
: Evaluates the computation using the given continuations. Promise are automatically evaluated (more on that below).
This makes code more logical, because the abstraction isn't making decisions for you. It also clarifies developer intent and allows for more descriptive error messages. Plus any utility written for Fantasy Land compatible types (like Ramda or Sanctuary) will also work on Futures.
It's worth noting that Futures (from Fluture) are considerably faster than Promises (around 20 times as fast in some of my tests), the points below will elaborate on this.
Promises are eager by design. When a new Promise is constructed its computation is immediately evaluated. When a Future is constructed, its computation is only evaluated once the continuations are provided. This has several implications for Promises:
- The computation runs immediately, before callbacks are available, so the resulting value has to be kept somewhere, making Promises inherently stateful.
- Any side-effects will run, even if continuations are never provided. This means Promises cannot be used to control side-effects.
- Errors might occur as a result of the evaluation, but Promises have no way to know whether they will be handled.
- Promises do a bunch of work in their constructors that Futures don't have to do because of this eager nature. There are performance costs associated with running operations that you don't necessarily need to run ;)
As mentioned above, Promises have to be stateful. This means that once it resolves to value or an error, it will stay in this new state. In other words; reevaluation becomes impossible. Every time you call .then
, the value is loaded from a cache.
Futures on the other hand are completely stateless by default. Every time you .fork
them, they reevaluate their computation. In most cases, you won't notice this difference. Promises and Futures are generally only forked once, allowing Futures to benefit from the performance gains of not having to keep track of state. If you want caching, it's quite easy to create a utility that does this for you. One such utility is Future.cache()
.
Promises are very involved in handling your errors. Errors thrown within continuations given to .then
are automatically caught and flow into the rejection branch at a very high performance cost. Ironically, Promises then allow you not to handle these errors by making handler
optional in .then(continuation, handler)
.
Fluture has a different approach: Thrown errors are never caught unless explicitly requested by the developer through functions like Future.try()
. The philosophy here is that if an error is thrown rather than passed into the rejection branch; it must be a developer mistake (aka. bug), and the program should crash and restart. A benefit of this behavior is that any errors which end up in the rejection branch of a Future are solely expected failures, meaning your program is less likely to enter an invalid state. More on this philosophy is very neatly described by @rpominovs document on exceptions.
Additionally, when a Future rejects the developer would have been forced to deal with it: handler
is not optional in .fork(handler, continuation)
, and as we've seen; the Future doesn't even run if .fork
wasn't called in the first place.