首页 > 编程语言 >JS异步编程

JS异步编程

时间:2023-06-27 16:22:39浏览次数:54  
标签:异步 resolve const log 编程 value JS reject console

ES5中的Generator异步方案

Generator使用

function* foo() {
  console.log('start')

  try {
    const res = yield 'foo'
    console.log(res) // bar
  } catch (e) {
    console.log(e)
  }
}

const generator = foo()
const res = generator.next()
console.log(res) // Object { value: "foo", done: false }

generator.next('bar')

// generator.throw(new Error('error'))

Gernerator配合Promise的异步方案

  • 基本实现
function* main() {
  const users = yield fetch('https://jsonplaceholder.typicode.com/users')
  console.log('users', users)

  const posts = yield fetch('https://jsonplaceholder.typicode.com/posts')
  console.log('posts', posts)
}

const g = main()
const res = g.next()

res.value.then(data => {
  const res2 = g.next(data) // data: users

  if (res2.done) return

  res2.value.then(data => {
    const res3 = g.next(data) // data: posts
  })
})
  • 封装成co函数
function* main() {
  try {
    const users = yield fetch('https://jsonplaceholder.typicode.com/users')
    console.log('users', users)

    const posts = yield fetch('https://jsonplaceholder.typicode.com/posts')
    console.log('posts', posts)

    const e = yield fetch('https://helloworld')
    console.log('e', e)
  } catch (e) {
    console.log('error', e)
  }
}

const co = generator => {
  const g = generator()

  const handleResult = result => {
    if (result.done) return
    result.value.then(
      data => {
        handleResult(g.next(data))
      },
      error => g.throw(error)
    )
  }

  handleResult(g.next())
}

co(main)

ES6中的async, await语法糖

async function main() {
  try {
    const users = await fetch('https://jsonplaceholder.typicode.com/users')
    console.log('users', users)

    const posts = await fetch('https://jsonplaceholder.typicode.com/posts')
    console.log('posts', posts)
  } catch (e) {
    console.log('error', e)
  }
}

const promise = main()
promise.then(() => {
  console.log('all done')
})

手写Promise

核心逻辑实现

const PENDING = 'pending' // 等待
const FULFILLED = 'fulfilled' // 成功
const REJECTED = 'rejected' // 失败

const resolvePromise = (promise, x, resolve, reject) => {
  if (promise === x) {
    // 防止自己返回自己
    return reject(new TypeError('Chaining cycle detected for promise #<Promise>'))
  }
  if (x instanceof MyPromise) {
    // promise 对象
    x.then(resolve, reject)
  } else {
    // 普通值
    resolve(x)
  }
}

class MyPromise {
  constructor(executor) {
    try {
      executor(this.resolve, this.reject)
    } catch (e) {
      this.reject(e)
    }
  }

  status = PENDING
  value = undefined
  reason = undefined
  successCallback = []
  failCallback = []

  resolve = value => {
    if (this.status !== PENDING) return // 状态一旦改变就不能再更改
    this.status = FULFILLED
    this.value = value

    // 判断成功回调是否存在,存在就调用
    while (this.successCallback.length) this.successCallback.shift()()
  }
  reject = reason => {
    if (this.status !== PENDING) return
    this.status = REJECTED
    this.reason = reason

    // 判断失败回调是否存在,存在就调用
    while (this.failCallback.length) this.failCallback.shift()()
  }

  then = (successCallback, failCallback) => {
    successCallback = successCallback ? successCallback : value => value
    failCallback = failCallback
      ? failCallback
      : reason => {
          throw reason
        }

    let promise = new MyPromise((resolve, reject) => {
      if (this.status === FULFILLED) {
        // 成功
        setTimeout(() => {
          try {
            let x = successCallback(this.value)
            resolvePromise(promise, x, resolve, reject) // 实现then方法的链式调用
          } catch (e) {
            reject(e)
          }
        }, 0)
      } else if (this.status === REJECTED) {
        // 失败
        setTimeout(() => {
          try {
            let x = failCallback(this.reason)
            resolvePromise(promise, x, resolve, reject) // 实现then方法的链式调用
          } catch (e) {
            reject(e)
          }
        }, 0)
      } else {
        // 异步等待
        this.successCallback.push(() => {
          setTimeout(() => {
            try {
              let x = successCallback(this.value)
              resolvePromise(promise, x, resolve, reject) // 实现then方法的链式调用
            } catch (e) {
              reject(e)
            }
          }, 0)
        })
        this.failCallback.push(() => {
          setTimeout(() => {
            try {
              let x = failCallback(this.reason)
              resolvePromise(promise, x, resolve, reject) // 实现then方法的链式调用
            } catch (e) {
              reject(e)
            }
          }, 0)
        })
      }
    })
    return promise
  }
}

let promise = new MyPromise((resolve, reject) => {
  setTimeout(() => {
    resolve('成功')
  }, 2000)
  // resolve('成功')
  // reject('失败')
})

promise
  .then()
  .then()
  .then(value => {
    console.log(value)
    return 'aaa'
  })
  .then(value => {
    console.log(value)
  })

Promise.all的实现

static all = array => {
    let result = []
    let index = 0

    return new MyPromise((resolve, reject) => {
      function addData(key, value) {
        result[key] = value
        index++
        if (index === array.length) {
          resolve(result)
        }
      }

      for (let i = 0; i < array.length; i++) {
        let current = array[i]
        if (current instanceof MyPromise) {
          // promise 对象
          current.then(
            value => addData(i, value),
            reason => reject(reason)
          )
        } else {
          // 普通值
          addData(i, array[i])
        }
      }
    })
  }

Promise.resolve的实现

static resolve = value => {
    if (value instanceof MyPromise) return value
    return new MyPromise(resolve => resolve(value))
  }

.finally、.catch方法的实现

finally = callback => {
    return this.then(
      value => {
        return MyPromise.resolve(callback()).then(() => value)
      },
      reason => {
        return MyPromise.resolve(callback()).then(() => {
          throw reason
        })
      }
    )
  }

catch = failCallback => {
	return this.then(undefined, failCallback)
}

标签:异步,resolve,const,log,编程,value,JS,reject,console
From: https://www.cnblogs.com/19BigData/p/17509200.html

相关文章

  • js常用操作
    1.重新渲染页面元素this.$forceUpdate(); 2.对象转化为json字符串再传参JSON.stringify()  3.删除js对象元素index表示对象下标deletethis.imgList[index];......
  • C语言中的网络编程:套接字和网络通信
    网络编程在C语言中是一个非常重要的主题。在这篇博客中,我们将深入了解(socket)和网络通信的基本概念。套接字是实现网络通信的关键部分,它允许计算机之间通过网络进行数据交换。我们将探索如何创建套接字、建立连接、发送和接收数据等操作。首先,我们需要包含一些头文件,这些头文件包含......
  • Linux系统编程21-简单的more命令实现
    /dev/tty:键盘和显示器设备描述文件向该文件写->显示在用户屏幕向该文件读->从键盘获取用户输入当more需要用户输入可以从/dev/tty得到数据#include<stdio.h>#include<stdlib.h>#definePAGELEN24#defineLINELEN512voiddo_more(FILE*);intsee_more(FILE*);......
  • java线程监控-jstack+jvisualvm
    Java线程监控一.Jstasck1.查找进程ps-ef|greptomcat-oa2.使用jstack监控jstack2429二、jvisualvm1.tomcat应用环境配置1.1tomcat环境配置修改tomcat中,catalina.sh文件cd/usr/src/tomcat-pinter/binvicatalina.sh在第二行添加如下:JAVA_OPTS="-Dcom.sun.mana......
  • JS 模拟 循环队列
     LoopArray代码(基于JS原生数组)/***循环队列*/varALoopQueue=(function(){/***@type{Array}*/letarr;/***头节点*@type{number}*/letfrontIdx;/***尾节点*@type{number}*/......
  • 线程池处理异步任务
    点击查看代码/***异步任务线程池(单例)*用于异步执行任务*/publicclassThreadPoolSingleton{privatestaticfinalAtomicReference<ThreadPoolSingleton>INSTANCE=newAtomicReference<>();privatefinalExecutorServiceexecutor;privateTh......
  • (Python编程)"添加Python,充分混和。"
    ProgrammingPython,3rdEdition翻译最新版本见:http://wiki.woodpecker.org.cn/moin/PP3eD23.1."AddPython.MixWell.Repeat."23.1."添加Python,充分混和。"Inthepriorchapter,weexploredhalfofthePython/Cintegration......
  • (Python编程)集成的方式
    ProgrammingPython,3rdEdition翻译最新版本见:http://wiki.woodpecker.org.cn/moin/PP3eD22.2.IntegrationModes22.2.集成的方式ThelasttwotechnicalchaptersofthisbookintroducePython'stoolsforinterfacingtotheoutsideworldanddiscussbothit......
  • SAP UI5 应用里 /sap/ui/thirdparty/datajs.js 的作用
    SAPUI5是一个基于JavaScript的用户界面技术,用于构建企业级应用程序。它是一个成熟的开源框架,由SAP开发,致力于提供高质量、可扩展和易于维护的Web应用程序。SAPUI5应用程序使用一系列技术和库,其中之一就是/sap/ui/thirdparty/datajs.js。在本文中,我们将详细讨论datajs.......
  • Spring面向切面编程(AOP)
    在软件开发中,我们经常需要解决一些与业务逻辑无关的横切关注点(cross-cuttingconcerns),例如日志记录、性能监测、事务管理等。传统的面向对象编程方法往往将这些关注点与核心业务逻辑混杂在一起,导致代码的可维护性和可扩展性下降。为了解决这个问题,Spring框架引入了面向切面编程(AOP......