Touch fish sauce article statement: content guarantee original, pure technology dry goods to share exchanges, do not play advertising do not brag force.

Foreword: This article will be unlike most written Promise articles you’ll see. It won’t talk about Promises/A+ or Promise. Race/race. In this article, I’ll use a lot of object-oriented thinking and focus only on the core idea of Promises and their implementation, which will give you a more structured understanding of Promises.

All right, with that bullshit, let’s get to the point. In the case of promises, it is explained in MDN as follows: The Promise object is used to represent the final completion (or failure) of an asynchronous operation and its resulting value. In this article, I’ve defined it differently, taking object-oriented programming as a starting point: A Promise object is a container that can be used to store the state and data resulting from asynchronous operations.

Around this concept, I’ve organized the Promise object using a UML class diagram to give you a better understanding of what I mean by container, as shown below:

In fact, the above class diagram can only be used to describe a container object structure for managing the state and data of synchronized operation results. Because the actual Promise container object needs to support managing the result state and data generated by asynchronous operations, its class diagram needs to add two internal properties for staging callback functions (for reasons described below), and its class diagram looks like this:

All of the discussion and implementation of promises that follow will revolve around this diagram, and if you’re interested in following along, I hope you’ll spend a little more time understanding this diagram to better read the rest of the article.

Ok, then I will use object-oriented programming to achieve the steps of programming as an idea, divided into the following parts to achieve the purpose of handwritten Promise:

  • With the container concept as a starting point, the basic structure of the Promise object is realized.
  • Analyze the relationship between Promise container and asynchronous operation, and implement Promise constructor.
  • Clarify the writing methods of data in Promise container, and implement resolve and Reject methods of Promise.
  • Clarify the way of reading the data in the Promise container and realize the then method of Promise.
  • Add a requirement to the THEN method to support chained calls to facilitate the handling of asynchronous operation flows.

When it comes to a written interview scenario that requires handwritten promises, individuals will also follow the steps outlined above.

First, the container concept as a starting point, design and implement the basic structure of the Promise object

Before analyzing the answer, I want you to consider the following two questions based on the picture above:

  • What data does the Promise container hold?
  • How do I read and write data inside the Promise container?

In the past, I referred to container reads and writes as fill and fetch operations, but since the data in the Promise container can be filled once and fetched numerous times, just like disk data, I refer to these operations as reads and writes.

1. Data stored in the container (object properties)

  • State: Indicates the container state, which can be divided into three types, namely pending, fulfilled and Rejected
  • Value: Data in the fulfilled state of the container
  • Reason: Data in the container Rejected state
  • OnResolvedTodoList: An array of callback behaviors in the fulfilled state, which is temporarily stored in the pending state and consumed after the fulfilled state
  • OnRejectedTodoList: An array of the callback behavior of the rejected state. It is stored pending and consumed after the Rejected state

In order to make the variable names more uniform and recognizable, we will replace the pity state with the resolved state in the following handwritten Promise implementation (just change the name, so there is no need to worry).

2. Read and write data in containers (object methods)

  • Write data: Inside the container, define and expose resolve and Reject methods that set internal state and data for external calls to write data (state, value/Reason) to the container.
  • Read Data: A then method that reads internal state and data is defined and exposed inside the container for external calls to read data in the container and trigger incoming callback behavior based on the container state.

3. Container object code implementation

After understanding the encapsulation composition (properties, methods) of the Promise object, we made the following code implementation based on the UML class diagram above:

Tips: Please make sure you understand the following code structure. This is the big picture of Promise.

class Container {
    state = undefined;
    value = undefined;
    reason = undefined;
    onResolvedTodoList = [];
    onRejectedTodoList = [];
    
    constructor(excutor) { // construct the container}
    
    resolve = value= > { // Write container data}
    reject = reason= > { // Write container data}
    
    then(onResolved, onRejected) { // Read the container data}
}

Container.PENDING = 'pending';
Container.RESOLVED = 'resolved';
Container.REJECTED = 'rejected';
Copy the code

To highlight the Container concept, I’ll give the Promise class we’ll write by hand a Container class name.

Two: Analyze the relationship between Promise container and asynchronous operation, and realize the constructor of Promise

Before we implement the Promise constructor, let’s examine the relationship between a wave of Promise containers and asynchronous operations:

  • Promise container: It is an asynchronous operation container that stores data and manages its read and write.
  • Asynchronous operations: a combination of synchronous and asynchronous statements that is our real code logic.
  • The relationship between Promise containers and asynchronous operations: Asynchronous operations entrust their execution results to Promise container objects for management.

The term trust is a good way to describe the relationship between asynchronous operations and the Promise container.

1. Use the term trust to understand the connection between asynchronous operations and promises

From the UML class diagram, we illustrated the Promise object structure, and now we fully understand the connection between asynchronous operations and the Promise container, centering on the entrust action:

  • Establish a trust: The external intention, the external selection Promise container manages the outcome of an asynchronous operation. (PS: ES6 is a callback function before a shuttle).
  • Entrust data to the container: The external will call the resolve and Reject methods in an asynchronous operation to determine what state and data to entrust to the container. It is important to note that we often entrust data in the callback of an asynchronous statement.
  • Request data from the container: External intentions, external decisions about when to request data and what to do after requesting data. It is important to note that the entrusted data is often asynchronous when the external reads the data in the container, so it is easy to cause the timing problem of the container data read and write.

From a requirement perspective, you can design your own Promise based on the idea of managing the results of asynchronous operations with container objects, and a hundred people will write a hundred promises.

Well, with the core understanding that the results of asynchronous operations are externally entrusted to the Promise container for management, we can understand the design logic of official Promises. Next, we implement the constructor of the official Promise container, which involves two processes: asynchronous operation, Promise commitment and data commitment to the container.

As for why official promises are designed this way, it’s a matter of requirement. We created the container to manage asynchronous results, so it implements both processes in the constructor (although our own promises may not be like this, But it does work, but I’m sure it’s not as friendly as the official version in terms of ease of use).

2. Implement the Promise constructor

(1): code implementation example

class Container {
    state = undefined;
    value = undefined;
    reason = undefined;
    onResolvedTodoList = [];
    onRejectedTodoList = [];
    
	// Take the excutor function as a construction parameter and call it immediately, passing the argument according to the excutor parameter convention.
    constructor(excutor) {
        try {
            excutor(this.resolve, this.reject);
            this.state = Container.PENDING;
        } catch (e) {
            this.reject(e)
        }
    }

    resolve = value= > { // Write container data}
    reject = reason= > { // Write container data}
    
    then(onResolved, onRejected) { // Read the container data}
}

Container.PENDING = 'pending';
Container.RESOLVED = 'resolved';
Container.REJECTED = 'rejected';
Copy the code

(2): Call construction example

// Externally define the excutor function with the convention parameters resolve and reject
const p1 = new Container((resolve, reject) = > {
  // The external determines when and what data is written to the container
  setTimeout(() = > {
    resolve(0)})})Copy the code

(3): Functional programming ideas see Excutor

The excutor function implements two functions:

  • Asynchronous operation carrier: function object form
  • Authorize external Settings to manage data and state in the container: container data write methods resolve and Reject

According to function responsibilities, analyze its function input and output to deepen understanding and impression:

  • Function input: Methods for writing data into the container, resolve and Reject
  • Function output: in the form of side effects, write container data
  • Mapping logic: During the execution of an asynchronous operation, data is written to the container as required by the caller

Three: clarify the writing methods of data in the Promise container, and implement the resolve and reject methods of Promise

Writing to container data essentially means assigning values to container data by container methods. If it is just a simple assignment that nature no knowledge can be discussed, but in the official version of the Promise there are a few details need to pay attention to, first take a look at the imitation of the official version of the Promise implementation example:

1. Implementation examples and invocation examples

The official Promise divides the container data into two mutually exclusive values (value and reason) and corresponds to the fullfilled and rejected states, which we might not have done if we had designed it ourselves.

class Container {
	/ /... attr
    constructor(excutor) {} 

    resolve = value= > {
        if (this.state ! = Container.PENDING)return
        this.status = Container.RESOLVED;
        this.value = value;
        while (this.onResolvedTodoList.length) this.onResolvedTodoList.shift()() // get the first one
    }

    reject = reason= > {
        if (this.state ! = Container.PENDING)return
        this.status = Container.REJECTED;
        this.reason = reason;
        while (this.onRejectedTodoList.length) this.onRejectedTodoList.shift()()
    }
    
    then(onResolved, onRejected){}}Copy the code

With that wrapped, let’s look at the call example:

const p1 = new Promise((resolve, reject) = > {
  // Write data synchronously to the container
  // resolve(0)
  // Write data to the container asynchronously
  setTimeout(() = > {
    resolve(1)})})Copy the code

2. Question: Why do nothing when writing data in the non-pending state?

The asynchronous data managed by the Promise container is the result of the operation and should not change after the operation has finished. This corresponds strongly to its name. Promise means a Promise that data in this container can only be entrusted and trusted by outsiders if the data after operation does not change.

This also means that we can only call the Resolve and Rejected methods once when we use promises. When we call them repeatedly, only the first time is valid.

3. Question: Why do I need to consume the staging callback queue after writing data?

The consumer pending callback queue stores callback actions that are added to the container while it is in the pending state. In many cases, when the then method is called to read the state and data in the container, the container cannot respond to the callback immediately because the state and data have not been written. But promises are reliable. They Promise that once their status and data are settled, they will do whatever we assign them, and they won’t miss a thing.

4. Question: Why do resolve and reject need to be set as arrow functions?

I was asked this question by a friend who asked why the resolve and reject functions were required to be arrow functions instead of function declarations. In fact, it is not necessary to declare functions as arrow functions. The reason why it is set to arrow functions is that the caller cannot change the this pointer in any way when calling a function declared as arrow functions, even if the call, apply and bind functions are used.

So, the arrow is just a constraint on the outside to change this in resolve and Reject, forcing the convention that resolve and Reject should only be used to write to the current container object.

Four: clear the way of reading the data in the Promise container and realize the then method of Promise

To read data from a Promise container, we define a get method that returns data from the Promise container. As you can see from the name of the then method, it is more than just get data. It means then, which is what the asynchronous operation should do next. Let’s start with a simple then method to understand the concept of then methods:

const obj = {
	value: 1.then: function (fn) {
    	fn(this.value)
    }
} 
obj.then(console.log);
Copy the code

For those of you who have studied functors, compare the then method of Promise with the functor map method. Think about why functors are good for data conversion flows and promises are good for asynchronous operation flows.

A thoughtful friend might point out that the callback argument in promise’s then function executes in a microtask-like fashion. I’m glad you asked that question. Yes, the then method in the above example is obviously synchronous and does not meet the promise’s THEN method requirements. Let’s modify it to make its callback function arguments execute as microtasks.

const obj = {
  value: 1.then: function (fn) {
    process.nextTick(() = > {
      fn(this.value); }); }}; obj.then(console.log);
Copy the code

Tips: Process is an API in node environment, which cannot be used in BOM environment.

Ok, so let’s take a look at an example of how to implement a then method that mocks the official Promise without chain-calling support (supported in point 5).

1. Implementation examples and invocation examples

Under the condition of without support chain calls, then function function and its implementation is very simple, it is responsible for receiving a user given two callback parameters, then the container stored data will be according to the state of the container after ready to decide which callback function calls, and at the same time to container storage data as a parameter to this function is called. Before implementing a function, let’s examine the THEN method using functional programming ideas.

  • Then function input: onResolved/onRejected callback that receives value/ Reason data as arguments.
  • Then function output: In the form of side effects, a call to a callback function with an argument.
  • Then function mapping logic: determines whether the container is ready. If it is not ready, the callback parameter function is temporarily stored. If it is ready, the callback function is executed based on the state.

Here is an example implementation and call of a then function that is so simple and short that it is easy to understand:

Implementation example:

class Container {
  state = undefined;
  value = undefined;
  reason = undefined;
  onResolvedTodoList = [];
  onRejectedTodoList = [];

  constructor(excutor) {}

  resolve = value = >{}
  reject = reason = >{}

  then(onResolved, onRejected) {
    // question: Why do we default onResolved and onRejected?
    onResolved = onResolved ? onResolved: value= > value;
    onRejected = onRejected ? onRejected: reason= > {	// question: Why the onRejected callback is not reason => reason?
      throw reason
    };
    switch (this.state) {
    // Question: Why do we need to determine pending state? Why does the code logic in this state hold the callback function instead of calling it?
    case Container.PENDING:
      // Question: Why put the callback onResolved and onRejected in the staging queue wrapped in the arrow function and passed in instead of onResolved or onRejected?
      this.onResolvedTodoList.push(() = >{
        // Question: The then function callback argument is not called in the form of micro task, why you write it in the form of macro task?
        setTimeout(() = > {
          // Question: Why is the callback function called without exception handling?
          onResolved(this.value);
        });
      });
      this.onRejectedTodoList.push(() = >{
        setTimeout(() = > {
          onRejected(this.reason);
        });
      });
      break;
    case Container.RESOLVED:
      setTimeout(() = > {
        onResolved(this.value);
      });
      break;
    case Container.REJECTED:
      setTimeout(() = > {
        onRejected(this.reason);
      });
      break; }}}Copy the code

Call example:

p1.then((value) = > {
  console.log(value)
}, (reason) = > {
  console.log(reason)
})
Copy the code

Let’s take a look at some of the core logic and issues in the code.

2. Q: Why do we default onResolved and onRejected parameters?

Personally, I think it is mainly for the following two aspects:

  • Avoid heavy null-detection logic in onResolved and onRejected callback calls.
  • In chained calls, the result state and data can be passed through without passing either or both callbacks. (Reasons explained in # 5)

3. Question: Why does the onRejected callback default processing logic is not reason => Reason?

Throw reason => {throw reason} => {throw reason} => {throw reason} Because promises are used to manage the outcome of an operation, which often means success or failure, we often treat the resolve state as success and the reject state as failure to read or write data. And when we execute a piece of synchronous or asynchronous operation code logic, sometimes we don’t need to pay attention to its successful value, but we do need to know about the failure of the operation, because it means that the operation entrusted to it may not succeed and may not have the side effects that we want it to have.

The Reason and reject states do not necessarily represent success or failure, and the corresponding data does not necessarily represent success or failure. A promise is simply a container in which the meaning of the managed state and data is always determined by the caller.

4. Question: Why do we need to determine pending state, and why does the code logic in this state hold the callback instead of calling it?

The pending state of Promise is designed to solve the problem of asynchronous operations causing the container to read and write data out of order. This pending state can be called the constructed but not ready state of the container. In this case, the state inside the container is pending and the data is undefined. When the then method of the Promise container is called to perform the next operation, the external party cannot get the state and data of the previous operation stored in the container. So it can’t be called.

To be sure, promise doesn’t invalidate callbacks that are added externally at this point in time, though they can’t be called immediately. Instead, it stores them temporarily and waits for them to be consumed when the container is ready. Consumption is represented by traversal ordered calls to onResolvedTodoList and onResolvedTodoList arrays in resolve and Reject.

5. Question: Why put the callback functions onResolved and onRejected in the staging queue and pass them in with the arrow function instead of onResolved or onRejected?

First of all, we need to clarify the requirement: the callback function needs to be called with value or Reason in the transient queue traversal, which is the execution of the callback function with parameters. Why not pass it through a call inside resolve or Reject? I don’t see anything wrong with doing that here.

If you have different views on this issue, please let me know in the comments section, thank you!

Whether to wrap a closure with arrow functions can be considered from the following three aspects:

  • When do you want this callback to be executed?
  • In what scope do you want this callback to be executed?
  • What do you want this in this callback to point to? (The function this refers to a function call.)

6. Question: Is the callback function parameter of then function called in the form of micro task? Why do you write it in the form of macro task?

I’m glad you asked that question. You’re right that, as the official Promise does, we really shouldn’t use setTimeout to register as a macro task to execute the then callback argument. However, there is no alternative. If we want our handwritten promises to be used in the BOM environment, we cannot use the process.nextTick method, and there is no suitable microtask API that can be used in the BOM environment. So I’ve written this and the following examples in the form of macro tasks (handwritten Promises used in the Node environment don’t have this problem). It’s not a big deal. Feel free to use setTimeout when writing promises, and you’ll be able to give an answer when the interviewer asks why.

Some might wonder why the official Promise can do this, since its then method’s callback argument is called in the form of a microtask. I can’t help it. Promises are built-in classes (or objects, depending on your understanding) that aren’t written in JavaScript, so they’re not bound by JavaScript or bom.

If this article does not get your likes, I will quit the nuggets!

7. Question: Why is the callback function called without exception handling?

Some friends may have this problem after seeing the implementation of the chain then method. What I want to say is that we don’t do exception handling just for the sake of doing it. Only do exception handling when it is truly worthwhile. What difference does it make whether we do exception handling here or not? It is the same to catch the call exception of this callback function here or globally. We don’t need to tell the next operation when we run into an exception like we did in chain, the next operation has a problem, so the next operation should go through the error callback.

If you have any suggestions on this, please let me know, thank you!

Five: add a requirement to then method, support chain call, convenient processing asynchronous operation flow

Promise’s greatest appeal is that it shifts the implementation of asynchronous flow of operations from callback nested coding to then “synchronous invocation” coding, eliminating the callback hell that has plagued front-end developers for years.

A quick reference to the asynchronous operation flow I was talking about, operation flow can be understood as operation -> next step -> next step -> next step ->… An asynchronous operation flow means that some operations in the above operation flow have asynchronous logic. Image point said, is that we through a number of then methods to put together the production workshop (you product, you fine product).

The chain-call nature of THEN allows Promise to handle the flow of asynchronous operations. If you understand THE chain-call nature of THEN from the requirements side, I recommend that you always think of it as the flow of asynchronous operations.

If I had to put a few labels on understanding Promise, I’d put three on it: container, asynchronous, and operation flow.

Return a new Promise object in the then method. There are a few key points to make here, though, so let’s look at an example implementation and invocation of the chained THEN method.

1. Chain then method implementation example and call example

Tips: It is recommended to compare the implementation example of the then method without chain invocation. In the process of comparison, find out the changes of the THEN method in order to implement the chain invocation.

class Container {
    constructor(excutor) {}

    resolve = value= > {}
    reject = reason= > {}

    onResolvedTodoList = [];
    onRejectedTodoList = [];

    then(onResolved, onRejected) {
        onResolved = onResolved ? onResolved : value= > value;
        onRejected = onRejected ? onRejected : reason= > { throw reason };
        let containerBack = new Container((resolve, reject) = > {
            switch (this.state) {
                case Container.PENDING:
                    this.onResolvedTodoList.push(() = > {
                        setTimeout(() = > {
                        	// Question: Why are exceptions handled here?
                            try {
                                const value = onResolved(this.value);
                                // Question: Why not just use the new value as container-managed data instead of wrapping a resolveContainer function?
                                resolveContainer(containerBack, value, resolve, reject);
                            } catch(e) { reject(e); }})});this.onRejectedTodoList.push(() = > {
                        setTimeout(function () {
                            try {
                                const value = onRejected(this.reason);
                                resolveContainer(containerBack, value, resolve, reject);
                            } catch(e) { reject(e); }})});break;
                case Container.RESOLVED:
                    setTimeout(() = > {
                        try {
                            const value = onResolved(this.value);
                            resolveContainer(containerBack, value, resolve, reject);
                        } catch(e) { reject(e); }})break;
                case Container.REJECTED:
                    setTimeout(function () {
                        try {
                            const value = onRejected(this.reason);
                            resolveContainer(containerBack, value, resolve, reject);
                        } catch(e) { reject(e); }})break; }});return containerBack
    }
}

function resolveContainer(containerBack, value, resolve, reject) {
    if(! (valueinstanceof Container)) {
      resolve(value)
    } else {
      if(value ! == containerBack) { value.then(resolve, reject); }else {
        reject(new TypeError('Chaining cycle detected for promise #<Promise>')); }}}Copy the code

Call example:

const p2 = p1.then((value) = > {
  console.log(value)
}, (reason) = > {
  console.log(reason)
})
p2.then((value) = > {
  console.log('p2', value)
}, (reason) = > {
  console.log('p2', reason)
}).then((value) = > {
  console.log('p3', value)
}, (reason) = > {
  console.log('p3', reason)
})
Copy the code

2. Question: Why exception handling here again?

That’s because in the case of chained calls, we can treat the internal operation of the callback function as a core part of the asynchronous operation hosted by the next promise (special cases in the next point), and if this new asynchronous operation fails, we consider the new promise to be in a reject state, And reason data is an exception object. Why not just not catch and let the outside report an error? This is because in the case of a chain-call, the user can handle the reject state and the data as a state, and promise has no right to interrupt.

3. Question: Why not just use the new value as container-managed data instead of wrapping a resolveContainer function?

Let’s go straight to the resolveContainer function:

function resolveContainer(containerBack, value, resolve, reject) {
    if(! (valueinstanceof Container)) {	// The callback returns a non-promise container object, which is essentially the same as passing the callback as an asynchronous operation when a new container is constructed
      resolve(value)
    } else {
      if(value ! == containerBack) { value.then(resolve, reject);The new Promise takes over the state and data of the Promise container returned by the callback function
      } else {	The new promise takes over the state and data of the new Promise container object (infinite dolls).
        reject(new TypeError('Chaining cycle detected for promise #<Promise>'))}}}Copy the code

It’s probably better to take a look at the above code and comments and figure it out for yourself. Why is this so? I understand it is the demand, we cannot simply put the callback function simply treated as the core part of the new promise asynchronous operations, this understanding is crucial, a lot of people is not a clear new promise what is the main body of asynchronous operation, and then it function callback function parameters of what is the relationship (you, You wonder how to construct the new Promise container object.

Any other questions? Feel free to ask questions in the comments section. If you can point out my problem, let me have a chance to correct, that would be even more thank you!