The origin of asynchronous programming

The execution environment of JavaScript language is single-threaded, only one task can be executed at a time, and multiple tasks need to be queued. This mode will block code, resulting in low efficiency of code execution, and will cause page lag and affect user experience. To avoid this problem, asynchronous programming is proposed; Asynchronous programming methods include: callback callback function, event listener, promise, etc.

JavaScript execution mode

  • Synchronous mode
    • Javascript code is executed sequentially, and the following code can be executed only after the previous code has been executed. In single thread, most code execution in synchronous mode, there are serious problems, the other one task execution time is too long, so the back of the task will be delayed, this delay is known as block, to the user will be the caton or getting stuck, asynchronous mode is required to solve program cannot avoid time-consuming operation of;
  • Asynchronous mode
    • Javascript code does not need to wait for the completion of the previous code execution to start, the subsequent logic of time-consuming tasks will generally be executed through the callback function, because a single thread can not process multiple files at the same time, the biggest difficulty of asynchronous single-thread mode is the chaotic sequence of code execution;
    • We put the code in the call stack to execute, if it is synchronous directly executed, if it is asynchronous code in the Queue after the synchronous code is finished;
    • EventLoop, which listens for tasks in the call stack and Queue
    • When all the tasks in the stack are finished, it will take a task from the Queue and put it in the stack, start executing, and wait for all the loops to end

Composition of asynchronous environment:

Web APIs environment, Queue message Queue, Call Stack Call stack: executing worksheet, Console print,EventLoop: listens on the Call Stack and Queue

  • Message Queue: A Queue is a place where asynchronous tasks are temporarily stored
  • Macro task
    • The code executing in the current call stack becomes a macro task; In order to ensure the orderly execution of JS macro tasks and DOM operations, the browser will re-render the page after the execution of one macro task and before the execution of the next macro task.
    • Macro tasks include setTimeout, setInterval, I/O, and UI interaction events
  • Micro tasks
    • Tasks that need to be executed immediately after the completion of the current task do not need to enter the Queue for execution. They are executed before the page rendering, which is faster than macro tasks.
    • Microtasks include: Promise, MutaionObserver, process.nexttick (‘ Node.js’) environment, etc

The browser execution engine first executes the Call Stack task, using the Event Loop to remove a task from the queue for execution. At any time during the task, we can forget to put some tasks in the message Queue, and these tasks will Queue up in a loop.

  • Callbacks: The foundation of all asynchronous programming schemes (think of it as something you want to do). Callers define functions that are handed to the executor, called callbacks.

Promise:

The CommonJS community came up with the Promise specification. Solve callback hell by flattening the nested callbacks as much as possible.

Basic use of Promise

A Promise is a global type. The constructor accepts a function argument and can accept two arguments (resolve, reject) inside the function. If it’s wrong, pass it reject. Once the status is determined, it cannot be modified.

The promise synchronization code completes before executing the promise.then() callback

Common mistakes:

Nested use is the wrong use of promise

By virtue of the chained invocation feature of the Promise then method, the asynchronous task should be as flat as possible.

Each THEN method actually adds a state-specific callback to the promise object returned by the previous THEN, and those promises are executed in turn. It is also possible to call back manually in the Promise object.

  • The then method of the Promise object returns a brand new Promise object
  • The subsequent THEN method registers the callback for the promise returned by the previous THEN method
  • The return value of the callback function from the previous THEN method is used as an argument to the callback from the later THEN method
  • If a promise is returned in the callback, then callbacks to subsequent then methods wait for it to end

Promise exception handling

Catch method registration failure callback

  • The promise fails or an exception executes the callback registered with onRejected.
  • The second parameter of then method, disadvantages: can not catch the round then method and the lower level of the exception;
  • Catch method: Recommended; Use a catch in the case of chain calls, because any exceptions in the promise chain are passed back, and the catch catches the exception

Promise static method

Sometimes you need to turn an existing object into a Promise object. The promise.resolve () method does this. Promise.resolve()

Resolve ('foo') // equivalent to new Promise(resolve => resolve('foo'))Copy the code
  • You can quickly convert normal values into Promise objects
  • If a promise is passed, the promise is returned as is
  • If the object is passed with the then method, the first parameter is onFulfilled and the second parameter is onFulfilled, which can also be used as a promise to continue the implementation. This object implements the thenable interface.

Promise parallel execution

  • Promise.all () waits for all tasks to end
    • Take an array
    • The array can be a Promise object or a normal value
    • This ends as long as all promise objects end successfully. And returns all promise instances
    • If either promise fails, the promise fails, returning the result of the first failed promise
  • promise.race()
    • As soon as one task completes, the promise value is returned

Promise execution timing

  • Microtasks improve overall responsiveness without waiting
  • Macro tasks: The tasks in Queue are called macro tasks. During the execution of macro tasks, some extra requirements can be added temporarily. The extra requirements can be selected as a new macro task and queued in the Queue. It can also be used as the [micro-task] of the current task to perform different queues immediately after the current task is finished.

Special: Promise callbacks are executed as microtasks, MutationObserver and process.nextTick as microtasks

Async/await syntactic sugar

Async is the syntactic sugar of a Generator function async returns a Promise object, and callbacks can be added using the then method. When a function executes, it returns as soon as it encounters await, waits for the asynchronous operation to complete, and then executes the following statement in the function body

async function main() {
    const user = await ajax('')
    console.log(user)
    const data = await ajax('')
    console.log(data)
}
Copy the code