Promise 笔记

来源:尚硅谷公开课

Promise

函数对象与实例对象

  1. 实例对象: new 函数产生的对象, 称为实例对象, 简称为对象
  2. 函数对象: 将函数作为对象使用(操作属性或方法)时, 简称为函数

一般来说,括号左边是函数,点的左边是对象(实例对象 or 函数对象)

a.b.c() //a.b.c是函数
a()[0]() //表示 a是个函数,返回值是个数组,返回的数组的第一个元素是一个函数
function Fn() { } //Fn是函数
const fn = new Fn() //Fn是构造函数,fn是实例对象(简称为对象)
console.log(Fn.protype) //Fn是函数对象
$('#test') //$是函数
$.get('/test') //$是函数对象调用get方法

二种类型的回调函数

什么是回调函数

  1. 自己定义的
  2. 自己没有主动执行
  3. 函数最终执行了
  • 同步回调

    ​ 立即执行, 完全执行完了才结束, 不会放入回调队列中

    const arr = [1, 3, 5]
    arr.forEach(item => { // 遍历回调,同步回调函数,不会放入队列,一上来就要执行完
        console.log(item)
    })
    console.log('----')
    // 输出结果为:1 3 5 ----
    
  • 异步回调

    ​ 不会立即执行, 会放入回调队列中将来执行

    setTimeout(() => { // 放入回调队列,异步回调函数
        console.log('timeout')
    }, 0) // 延迟时间为0
    console.log('timeout之后')
    // 输出结果为:timeout之后 timeout
    

如何判断回调函数是同步的还是异步的

在回调函数之后使用console.log(),看执行顺序,参考上述。

常见的内置错误

错误的类型

​ Error: 所有错误的父类型

  1. ReferenceError: 引用的变量不存在(常见)

    console.log(a) // ReferenceError: a is not defined
    console.log('----') // 没有捕获error,下面的代码不会执行
    
  2. TypeError: 数据类型不正确的错误(常见)

    let b
    b.xxx() // TypeError: Cannot read property 'xxx' of undefined
    console.log(b.xxx) // TypeError: Cannot read property 'xxx' of undefined
    
    let c = {}
    c.xxx() // TypeError: c.xxx is not a function
    
  3. RangeError: 数据值不在其所允许的范围内

    function fn() {
        fn() // 递归调用
    }
    fn() // RangeError: Maximum call stack size exceeded
    
  4. SyntaxError: 语法错误

    const d = """" // SyntaxError: Unexpected string
    

错误的处理(捕获与抛出)

错误对象

message属性: 错误相关信息

stack属性: 函数调用栈记录信息

错误处理

  • 捕获错误: try…catch

    try { // 将可能有错误的代码放入 try{ } 中
        let d
        console.log(d.xxx)
    } catch (error) { // 处理错误,其中,error是一个对象,有两个属性,如下图
        console.log(error.message)
        console.log(error.stack)
    }
    

在这里插入图片描述

  • 抛出错误: throwError

    function something() {
      if (Date.now()%2 === 1) {
        console.log('当前时间为奇数')
      } else { // 如果时间是偶数则抛出异常,由调用来处理
        throw new Error('当前时间为偶数') // 一般都只需要直接抛出Error,不需要细分
      }
    }
    
    // 捕获处理异常
    try {
      something()
    } catch (error) { 
      alert(error.message)
    }
    

如何使用Promise

Promise的理解

Promise是异步编程的一种解决方案

如果网络请求是同步的,则在请求时,下面的代码会被阻塞而不运行

封装一个网络请求的函数,因为不能立即拿到结果,所以不能像简单的3+4=7一样将结果返回

所以往往我们会传入另外一个函数,在数据请求成功时,将数据通过传入的函数回调出去

如果只是一个简单的网络请求,那么这种方案不会给我们带来很大的麻烦

但是,当网络请求非常复杂时,就会出现回调地狱

我们需要通过一个url1从服务器加载一个数据data1,data1中包含了下一个请求的url2

我们需要通过data1取出url2,从服务器加载数据data2,data2中包含了下一个请求的url3

我们需要通过data2取出url3,从服务器加载数据data3,data3中包含了下一个请求的url4

发送网络请求url4,获取最终的数据data4
在这里插入图片描述
这样的代码难看而且不容易维护,我们期望一种更加优雅的方式来进行这种异步操作

Promise可以以一种非常优雅的方式来解决这个问题

Promise的作用

抽象表达:

​ Promise 是 JS 中进行异步编程的新的解决方案

具体表达:

​ (1) 从语法上来说:Promise 是一个构造函数

​ (2) 从功能上来说:Promise 对象用来封装一个异步操作并可以获取其结果

什么情况下会用到Promise?

​ 一般情况下是有异步操作时,使用Promise对这个异步操作进行封装

​ Promise将 网络请求的代码 和 相关的处理代码 进行了分离

为什么要用Promise

  1. 指定回调函数的方式更加灵活

​ 旧的:回调函数必须在启动异步任务前指定

​ promise:启动异步任务 ==> 返回Promise对象 ==> 给promise对象绑定回调函数(甚至可以在异步任务结束后指定)

  1. 支持链式调用,解决回调地狱问题

​ 回调地狱:回调函数嵌套调用,外部回调函数异步执行的结果是嵌套的回调函数执行的条件

​ 回调地狱的缺点:不便于阅读 / 不便于异常处理

​ 解决方案:promise链式调用(Promise中还有回调函数,因此不是最终方案)

​ 终极解决方案:async / await

// 成功的回调函数
function successCallback(result) {
  console.log("声音文件创建成功: " + result);
}

// 失败的回调函数
function failureCallback(error) {
  console.log("声音文件创建失败: " + error);
}

/* 1.1 使用纯回调函数 */
createAudioFileAsync(audioSettings, successCallback, failureCallback) // 先创建回调函数,后调用

/* 1.2 使用Promise */
const promise = createAudioFileAsync(audioSettings);
setTimeout(() => {
  promise.then(successCallback, failureCallback);
}, 3000);

/* ------------------------------------ */

/* 2.1 回调地狱 */
doSomething(function(result) {
  doSomethingElse(result, function(newResult) {
    doThirdThing(newResult, function(finalResult) {
      console.log('Got the final result: ' + finalResult)
    }, failureCallback)
  }, failureCallback)
}, failureCallback)

/* 2.2 使用promise的链式调用解决回调地狱 */
doSomething()
  .then(function(result) {
  return doSomethingElse(result)
})
  .then(function(newResult) {
  return doThirdThing(newResult)
})
  .then(function(finalResult) {
  console.log('Got the final result: ' + finalResult)
})
  .catch(failureCallback)

/* 2.3 async/await: 回调地狱的终极解决方案 */
async function request() {
  try {
    const result = await doSomething()
    const newResult = await doSomethingElse(result)
    const finalResult = await doThirdThing(newResult)
    console.log('Got the final result: ' + finalResult)
  } catch (error) {
    failureCallback(error)
  }
}

Promise的状态改变

  1. pending 变为 resolved

  2. pending 变为 rejected

    说明: 只有这 2 种,且一个 promise 对象只能改变一次,无论变为成功还是失败,都会有一个结果数据,成功的结果数据一般称为 value,失败的结果数据一般称为 reason

Promise的基本流程

Promise的基本使用

/* 简单示例 */
new Promise((resolve, reject) => {
    setTimeout(() => { // 网络请求
      if (异步操作成功) {
        resolve('this is data') // 成功的时候调用resolve
      }else if (异步操作失败) {
        reject('error message') // 失败的时候调用reject
      }else {
        ...
      }
    }, 1000)
  }).then(
    data => {
      // 100行的处理代码
      console.log(data)
      console.log(data)
    },
    err => {
      // 100行的处理代码
      console.log(err)
      console.log(err)
    })
    // .catch(err => { // 如果then中不处理error,则可以用catch专门处理error
    //   // 100行的处理代码
    //   console.log(err)
    // })
//1.创建一个新的promise对象
const p = new Promise((resolve, reject) => {//执行器函数
  // 2.执行异步操作任务
  setTimeout(() => {
    const time = Date.now() // 如果当前时间是偶数就代表成功,否则代表失败
    if(time%2 == 0){
      // 3.1. 如果成功了,调用resolve(value)
      resolve('成功的数据,time=' + time)
    } else {
      // 3.2. 如果失败了,调用reject(reason)
      reject('失败的数据,time=' + time)
    }
  }, 1000);
})

p.then(
  value => { // 接收得到成功的value数据  onResolved
  	console.log( '成功的回调',value)
  },
  reason => { // 接收得到失败的reason数据  onRejected
  	console.1og( '失败的回调',value)
  }
)

/* ------------------------------------ */

链式调用的简写

/* Promise的链式调用 */

  // 网络请求: aaa -> 自己处理(10行)
  // 处理: aaa111 -> 自己处理(10行)
  // 处理: aaa111222 -> 自己处理

  /* 未简写 */
  new Promise((resolve, reject) => {
    setTimeout(() => {
      resolve('aaa')
    }, 1000)
  }).then(res => {
    // 1.自己处理10行代码
    console.log(res, '第一层的10行处理代码');
  
    // 2.对结果进行第一次处理
    return new Promise(resolve => { // 可以简写*******************
      resolve(res + '111')
    })
  }).then(res => {
    console.log(res, '第二层的10行处理代码');
  
    return new Promise(resolve => {
      resolve(res + '222')
    })
  }).then(res => {
    console.log(res, '第三层的10行处理代码');
  })

/* ---------------------------------------------------------------- */

  /* 简写 */
  // new Promise(resolve => resolve(结果)) 简写为 return Promise.resolve(res + '111')
  new Promise((resolve, reject) => {
    setTimeout(() => {
      resolve('aaa')
    }, 1000)
  }).then(res => {
    // 1.自己处理10行代码
    console.log(res, '第一层的10行处理代码');

    // 2.对结果进行第一次处理
    return Promise.resolve(res + '111') // 简写*******************
  }).then(res => {
    console.log(res, '第二层的10行处理代码');

    return Promise.resolve(res + '222')
  }).then(res => {
    console.log(res, '第三层的10行处理代码');
  })

/* ---------------------------------------------------------------- */

  /* 继续简写 */
  // 省略掉Promise.resolve
  new Promise((resolve, reject) => {
    setTimeout(() => {
      resolve('aaa')
    }, 1000)
  }).then(res => {
    // 1.自己处理10行代码
    console.log(res, '第一层的10行处理代码');

    // 2.对结果进行第一次处理
    return res + '111' // 继续简写,省略掉Promise.resolve,内部会进行Promise包装*******************
  }).then(res => {
    console.log(res, '第二层的10行处理代码');

    return res + '222'
  }).then(res => {
    console.log(res, '第三层的10行处理代码');
  })

/* ---------------------------------------------------------------- */

/* 同理,reject也可以进行简写 */
// 未简写
...
.then(res => {
  return new Promise((resolve, reject) => {
    // resolve('aaa')
    reject('err')
	})
})
...
	.catch(err => {
    console.log(err)
  })
/* 可以简写为:*/
...
.then(res => {
  return Promise.reject('error message')
})
...
	.catch(err => {
    console.log(err)
  })
/* 也可以简写为:*/
...
.then(res => {
  throw 'error message'
})
...
	.catch(err => {
    console.log(err)
  })

三种状态

pending:等待状态,比如正在进行网络请求,或者定时器没有到时间。

fulfill:满足状态,当我们主动回调了resolve时,就处于该状态,并且会回调.then()

reject:拒绝状态,当我们主动回调了reject时,就处于该状态,并且会回调.catch()

在这里插入图片描述

相关API

  1. Promise构造函数: Promise (excutor) {}

​ excutor函数: 同步执行 (resolve, reject) => {}

​ resolve函数: 内部定义成功时我们调用的函数 value => {}

​ reject函数: 内部定义失败时我们调用的函数 reason => {}

​ 说明: excutor会在Promise内部立即同步回调,异步操作在执行器中执行

  1. Promise.prototype.then方法: (onResolved, onRejected) => {}

​ onResolved函数: 成功的回调函数 (value) => {}

​ onRejected函数: 失败的回调函数 (reason) => {}

​ 说明: 指定用于得到成功value的成功回调和用于得到失败reason的失败回调,返回一个新的promise对象

Promise的回调函数是异步执行的

  1. Promise.prototype.catch方法: (onRejected) => {}

​ onRejected函数: 失败的回调函数 (reason) => {}

​ 说明: then()的语法糖,相当于: then(undefined, onRejected)

  1. Promise.resolve方法: (value) => {}

​ value: 成功的数据或promise对象

​ 说明: 返回一个成功/失败的promise对象

  1. Promise.reject方法: (reason) => {}

​ reason: 失败的原因

​ 说明: 返回一个失败的promise对象

  1. Promise.all方法: (promises) => {}

​ promises: 包含n个promise的数组

​ 说明: 返回一个新的promise,只有所有的promise都成功才成功,只要有一个失败了就直接失败

  1. Promise.race方法: (promises) => {}

​ promises: 包含n个promise的数组

​ 说明: 返回一个新的promise,第一个完成的promise的结果状态就是最终的结果状态

all 和 race 方法

Promise.all([
    new Promise((resolve, reject) => {
      setTimeout(() => {
        resolve({name: 'why', age: 18})
      }, 2000)
    }),
    new Promise((resolve, reject) => {
      setTimeout(() => {
        resolve({name: 'kobe', age: 19})
      }, 1000)
    })
  ]).then(results => {
    console.log(results);
  })

Promise.race([
    new Promise((resolve, reject) => {
      setTimeout(() => {
        resolve({name: 'why', age: 18})
      }, 2000)
    }),
    new Promise((resolve, reject) => {
      setTimeout(() => {
        resolve({name: 'kobe', age: 19})
      }, 1000)
    })
  ]).then(
  	 // 输出的是先执行完成的
     value => {
        console.log(value)
     },
     reason => {
        console.log(reason)
     })
new Promise((resolve, reject) => {
    setTimeout(() => {
      // resolve('成功的数据') // resolve和reject只能选之一,因为pending只能改变一次
      reject('失败的数据')
    }, 1000)
  }).then(
    value => {
      console.log('onResolved()1', value)
    }
  ).catch(
    reason => {
      console.log('onRejected()1', reason) // onRejected()1 失败的数据
    }
  )

   // 产生一个成功值为1的promise对象
   const p1 = new Promise((resolve, reject) => {
     setTimeout(() => {
       resolve(1)
     }, 100);
   })
   const p2 = Promise.resolve(2)
   const p3 = Promise.reject(3)
   p1.then(value => {console.log(value)}) // 1
   p2.then(value => {console.log(value)}) // 2
   p3.catch(reason => {console.log(reason)}) // 3

   const pAll = Promise.all([p1, p2, p3]) // p3为reject,失败,所以all方法为失败
   const pAll = Promise.all([p1, p2]) // p1、p2都为resolve,成功,所以all方法为成功
   pAll.then(
     // 输出的顺序与all方法的形参中的顺序一致,与执行的顺序无关
     values => {
        console.log('all onResolved()', values) 
     },
     reason => {
        console.log('all onRejected()', reason)
     }
   )
 
   const pRace = Promise.race([p1, p2, p3])
   pRace.then(
     // 输出的是先执行完成的
     value => {
        console.log('race onResolved()', value)
     },
     reason => {
        console.log('race onRejected()', reason)
     }
   )

关键问题

  1. 如何改变 promise 的状态?

    (1) resolve(value):如果当前是 pending 就会变为 resolved

    (2) reject(reason):如果当前是 pending 就会变为 rejected

    (3) 抛出异常:如果当前是 pending 就会变为 rejected

  2. 一个 promise 指定多个成功/失败回调函数,都会调用吗?

    当 promise 改变为对应状态时都会调用

const p = new Promise((resolve, reject) => {
  // resolve(1) // promise变为resolved成功状态
  // reject(2) // promise变为rejected失败状态
  // throw new Error('出错了') // 抛出异常,promse变为rejected失败状态,reason为抛出的error
  throw 3 // 抛出异常, promse变为rejected失败状态, reason为抛出的3
})
p.then(
  value => {},
  reason => {console.log('reason', reason)} // reason 3
)
p.then(
  value => {},
  reason => {console.log('reason2', reason)} // reason2 3
)

  1. 改变promise状态 和 指定回调函数 谁先谁后?

    (1)都有可能,正常情况下是先指定回调再改变状态,但也可以先改状态再指定回调

    (2)如何先改状态再指定回调?

    ​ ①在执行器中直接调用resolve()/reject()

    ​ ②延迟更长时间才调用then()

    (3)什么时候才能得到数据?

    ​ ①如果先指定的回调,那当状态发生改变时,回调函数就会调用,得到数据

    ​ ②如果先改变的状态,那当指定回调时,回调函数就会调用,得到数据

/* 常规:先指定回调函数, 后改变状态 */
new Promise((resolve, reject) => {
  setTimeout(() => {
    resolve(1) // 后改变状态(同时指定数据),异步执行回调函数
  }, 1000);
}).then(// 先指定回调函数,保存当前指定的回调函数,只有先保存之后才能执行回调函数
  value => {},
  reason => {console.log('reason', reason)}
)

/* 先改状态, 后指定回调函数 */
new Promise((resolve, reject) => {
  resolve(1) // 先改变状态(同时指定数据)
}).then( // 后指定回调函数, 异步执行回调函数
  value => {console.log('value2', value)},
  reason => {console.log('reason2', reason)}
)
console.log('-------')

/* 先改状态, 后指定回调函数 */
const p = new Promise((resolve, reject) => {
  setTimeout(() => {
    resolve(1) // 先改变状态(同时指定数据), 异步执行回调函数
  }, 1000); // 延迟1秒
})
setTimeout(() => {
  p.then( // 后指定回调函数
    value => {console.log('value3', value)},
    reason => {console.log('reason3', reason)}
  )
}, 1100); // 延迟1.1秒

  1. promise.then()返回的新promise的结果状态由什么决定?

    (1)简单表达: 由then()指定的回调函数执行的结果决定

    (2)详细表达:

    ​ ①如果抛出异常,新promise变为rejected,reason为抛出的异常

    ​ ②如果返回的是非promise的任意值,新promise变为 resolved / value 为返回的值

    ​ ③如果返回的是另一个新promise,此promise的结果就会成为新promise的结果

    在一个then()中是跳转到失败[reject],只表示上一个Promise是失败的,如果在这个then()中没有继续返回失败,则[如果有]下一个then是转到resolve的

    new Promise((resolve, reject) => {
      // resolve(1)
      reject(1)
    }).then(
      value => {
        // console.log('onResolved1()', value) //结果为:onResolved1() 1,对应第2行,然后转到第22行
        /* 如果不指定返回值,则默认返回undefined */
        return 2 // 此时第22行结果为 3
        // return Promise.resolve(3) // 此时第22行结果为 3
        // return Promise.reject(4) // 此时第25行结果为 4
        // throw 5 //此时第25行结果为 5
      },
      reason => {
        // console.log('onRejected1()', reason) //结果为:onRejected1() 1,对应第3行,然后转到第22行(因为这个reject是成功的)
        return 2 // 此时第22行结果为 2
        // return Promise.resolve(3) // 此时第22行结果为 3
        // return Promise.reject(4) // 此时第25行结果为 4
        // throw 5 //此时第25行结果为 5
      }
    ).then(
      value => {
        console.log('onResolved2()', value) 
      },
     reason => {
        console.log('onRejected2()', reason)
      }
    )
    
    
  2. promise如何串连多个操作任务(同步异步都可以)?

    (1)promise的 then() 返回一个新的promise,可以 then() 的链式调用

    (2)通过then的链式调用串连多个同步/异步任务

    在then()中:

    再执行同步任务,不需要return新的Promise;

    再执行异步任务,应当将其封装到Promise对象中

new Promise((resolve, reject) => {
  setTimeout(() => {
    console.log("执行任务1(异步)") // 第1个打印:执行任务1(异步)
    resolve(1)
  }, 1000);
}).then(
  value => {
    console.log('任务1的结果: ', value) // 第2个打印:任务1的结果: 1
    console.log('执行任务2(同步)') // 第3个打印:执行任务2(同步)
    return 2 // 在回调函数中再执行同步任务,不需要返回新的Promise
  }
).then(
  value => {
    console.log('任务2的结果:', value) // 第4个打印:任务2的结果: 2
		
    /* 在回调函数中再执行异步任务,应当将其封装到Promise对象中 */
    // 只有再返回一个Promise对象,第26行的then才是这个返回的Promise对象的
    return new Promise((resolve, reject) => {
      // 启动任务3(异步)
      setTimeout(() => {
        console.log('执行任务3(异步))') // 第5个打印:执行任务3(同步)
        resolve(3)
      }, 1000);
    })
  }
).then( // 这个then是第18行返回的Promise的
  value => {
    console.log('任务3的结果: ', value) // 第6个打印:任务3的结果: 3
  }
)

  1. Promise异常传/穿透?

    (1)当使用promise的then链式调用时, 可以在最后指定失败的回调,失败会一层层传递下去

    (2)前面任何操作出了异常, 都会传到最后失败的回调中处理,因为失败会一层层传递下去

  2. 中断promise链?

    (1)当使用promise的then链式调用时, 在中间中断, 不再调用后面的回调函数

    (2)办法: 在回调函数中返回一个pending状态的promise对象【return new Promise(() => {})】

new Promise((resolve, reject) => {
  // resolve(1)
  reject(1)
}).then(
  value => {
    console.log('onResolved1()', value)
    return 2
  }
  /* 没有指定reject默认相当于:reason => {throw reason} */
).then(
  value => {
    console.log('onResolved2()', value)
    return 3
	}
  /* 没有指定reject默认相当于:reason => {throw reason} */
).then(
  value => {
    console.log('onResolved3()', value)
  },
  /* 也可以是:reason => Promise.reject(reason) */
).catch(reason => { // 这里是catch
  console.log('onRejected1()', reason) // 跳转到第29行
  // throw reason // 跳转到32行
  // Promise.reject(reason) // 值是undefined,没有抛出异常不会跳转到32行,而是进入到29行
  // return Promise.reject(reason) // 跳转到32行
  return new Promise(() => {}) // 返回一个pending的promise,中断promise链,不会继续传递到下一个then,返回的新的Promise决定下一个then的跳转,如果新的Promise没有结果,则会中断
}).then(
  value => {
    console.log('onResolved3()', value)
  },
  reason => {
    console.log('onRejected2()', reason)
  }
)

async & await

async

MDN文档:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Statements/async_function

简单来说,在异步操作之前添加 async 。

async 表示函数里有异步操作,调用该函数会立即返回一个 Promise 对象。async 函数内部 return 语句返回的值,会成为 then 方法回调函数的参数。async 函数内部抛出错误会导致返回的 Promise 对象变为 reject 状态。抛出的错误对象会被 catch 方法回调函数接收到。

async function f() {
	return 'hello world'
}

f().then(v => console.log(v)) 
// "hello world"


async function f() {
  throw new Error('出错了')
}

f().then(
	v => console.log(v),
  e => console.log(e)
)
// Error: 出错了

await 表示紧跟在后面的表达式需要等待结果。async 函数返回的 Promise 对象必须等到内部所有 await 命令后面的 Promise 对象执行完才会发生状态改变,除非遇到 return 语句或者抛出错误。也就是说,只有 async 函数内部的异步操作执行完,才会执行 then 方法指定的回调函数。

async 函数的返回值是 Promise 对象,可以用then方法指定下一步的操作。进一步说,async 函数完全可以看作由多个异步操作包装成的一个 Promise 对象,而 await 命令就是内部 then 命令的语法糖。

async 函数返回一个 Promise 对象,可以使用 then 方法添加回调函数。当函数执行的时候,一旦遇到 await 就会先返回,等到异步操作完成,再接着执行函数体内后面的语句。

  1. 函数的返回值为 promise 对象;即使主动 return 其他值,async返回的也是 promise 对象。
  2. promise对象的结果由 async 函数执行的返回值决定。
/* async 函数的使用形式 */
// 函数声明
async function foo() { }

// 函数表达式
const foo = async function () { }

// 对象的方法
let obj = { async foo() { }}
obj.foo().then(...)
               
// Class 的方法
class Storage {
	constructor() {
  	this.cachePromise = caches.open('avatars')
	}

	async getAvatar(name) {
    const cache = await this.cachePromise
    return cache.match('/avatars/${name}.jpg')
  }
}

const storage = new Storage()
storage.getAvatar('jake').then(...)
                               
// 箭头函数
const foo = async () => { }
// 例子1
async function fn1() {
  return 1
}

async function fn2() {
  const value = await fn1() // value的值为:1
}
/*
	value的值为1并不是因为fn1返回了1,fn1()加了async声明因此返回的是promise对象且promise的值为1,
	然后通过await获取了promise对象的值,该值为1
*/
// 例子2
function timeout(ms) {
  return new Promise((resolve) => {
    setTimeout(resolve, ms);
  });
}

async function asyncPrint(value, ms) {
  await timeout(ms);
  console.log(value); // 50ms后输出 hello world
}

asyncPrint('hello world', 50);
// 例子3
async function getTitle(url) {
  let response = await fetch(url)
  let html = await response.text()
  return html.match(/<title>([\s\S+])<\/title>/i)[1]
}
getTitle('https://tc39.github.io/ecma262/').then(console.log)
// "ECMAScript 2017 Language Specification"
/* 
	上面的代码中,函数 getTitle 内部有3个操作:抓取网页、取出文本、匹配页面标题。只有这3个操作全部完成,才会执行 then 方法里面的 console.log
*/

await

MDN文档:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Operators/await

  1. await 右侧的表达式一般为 promise 对象,但也可以是其他的值
  2. 如果表达式是 promise 对象,await 返回的是 promise 成功的值
  3. 如果表达式是 其他值,则返回该值本身
  4. 函数中使用了await,则该函数必须声明为async(即,await必须用在async声明的函数中);但async声明的函数不一定要使用await
  5. await 只能得到成功的结果,如果await的promise失败,则会抛出异常;如果想要得到失败的结果,则需要使用 try … catch …

正常情况下,await 命令后面是一个 Promise 对象。如果不是,会被转成一个立即 resolve 的 Promise 对象。

async function f() {
	return await 123
}

f().then(v => console.log(v)) // 123
// 上面的代码中,await 命令的参数是数值123,它被转成 Promise 对象并立即 resolve

await 命令后面的 Promise 对象如果变为 reject 状态,则 reject 的参数会被 catch 方法的回调函数接收到。并且,只要一个 await 语句后面的 Promise 变为 reject,那么 async 函数都会中断执行;

如果不希望被中断执行,

① 可以将出错的 await 放在 try…catch 结构里,这样不管这个异步操作是否成功,后面的 await 都会执行。

② 在 await 后面的 Promise 对象添加一个 catch 方法,处理前面可能出现的错误。

async function f() {
	await Promise.reject('出错了')
  await Promise.reject('111')
  await Promise.reject('222')
}

f()
	.then(v => console.log(v))
	.catch(e => console.log(e))
// 出错了 111 222
/* 
上面的代码中,await 语句前面没有 return,但是 reject 方法的参数依然传入了catch 方法的回调函数。这里如果在 await 前面加上 return,效果是一样的。
*/

// 中断执行
async function f() {
  await Promise.reject('出错了') // Promise 变为 reject,中断执行
  await Promise.resolve('hello world') // 不会执行
}

// 方法一:采用 try...catch 结构
async function f() {
  try {
    await Promise.reject('出错了')
  } catch(e) {
    ...
  }
  return await Promise.resolve('hello world')  
}

f().then(v => console.log(v)) // hello world

// 方法二:Promise.reject后面添加一个 catch 方法
async function f() {
  await Promise.reject('出错了').catch(e => console.log(e)) // 出错了
  return await Promisse.resolve('hello world')
}

f().then(v => console.log) // hello world
// 使用 try...catch结构,实现多次重复尝试
const superagent = require('superagent')
const NUM_RETRIES = 3

async function test() {
  let i
  for(i=0; i<NUM_RETRIES; ++i) {
    try {
      await superagent.get('http://google.com/this-throws-an-error')
      break
    } catch(err) {...}
  }
  console.log(i) // 3
}

test()
// 上面的代码中,如果 await 操作成功,则会使用 break 语句退出循环;如果失败,则会被 catch 语句捕捉,然后进入下一轮循环。

JS异步之宏队列与微队列

在这里插入图片描述

  1. JS中用来存储待执行回调函数的队列包含 2 个不同特定的列队

  2. 宏列队:用来保存待执行的宏任务(回调),比如:定时器回调 / DOM事件回调 / ajax回调

  3. 微列队:用来保存待执行的微任务(回调),比如:promise的回调 / MutationObserver的回调

  4. JS执行时会区别这 2 个队列:

(1)JS引擎首先必须先执行所有的初始化同步任务代码,然后再执行异步代码

(2)在执行一个宏任务前,优先执行微任务(如果有),微任务顺序执行完毕后,再顺序执行宏任务,如果在执行新的宏任务前又有了新的微任务,则又将先顺序执行完毕微任务

【即,①优先执行同步任务;②异步任务中只要有微任务,则不会执行宏任务】

setTimeout(() => { // 会立即放入宏列队
  console.log('timeout callback1()')
  Promise.resolve(3).then(
    value => { // 会立即放入微列队
      console.log('Promise onResolved3()', value)
    }
  )
}, 0)
setTimeout(() => { // 会立即放入宏列队
  console.log('timeout callback2()')
}, 0)
Promise.resolve(1).then(
  value => { // 会立即放入微列队
    console.log('Promise onResolved1()', value)
  }
) 
Promise.resolve(2).then(
  value => { // 会立即放入微列队
    console.log('Promise onResolved2()', value)
  }
) 

// 执行结果为:
Promise onResolved1() 1
Promise onResolved2() 2
timeout callback1()
Promise onResolved3() 3
timeout callback2()

面试题

面试题1

setTimeout(()=>{
  console.log(1)
},0)
Promise.resolve().then(()=>{
  console.log(2)
})
Promise.resolve().then(()=>{
  console.log(4)
})
console.log(3)

/*
	输出结果为:3 2 4 1
	3 为同步
	then函数为异步
*/

面试题2

setTimeout(() => {
  console.log(1)
}, 0)
new Promise((resolve) => {
  console.log(2)
  resolve()
}).then(() => {
  console.log(3)
}).then(() => { // 这个then需要上面的then执行完毕才会执行,因此不会马上放入队列中
  console.log(4)
})
console.log(5)

/*
	输出结果为:2 5 3 4 1
	2 5 为同步
*/

面试题3

const first = () => (new Promise((resolve, reject) => {
  console.log(3)
  let p = new Promise((resolve, reject) => {
    console.log(7)
    setTimeout(() => {
      console.log(5)
      resolve(6)
    }, 0)
    resolve(1)
  })
  
  resolve(2)
  
  p.then((arg) => {
    console.log(arg)
  })

}))

first().then((arg) => {
  console.log(arg)
})
console.log(4)

/*
	输出结果为:3 7 4 1 2 5
	3 7 4 为同步
*/

面试题4

setTimeout(() => {
  console.log("0")
}, 0)
new Promise((resolve,reject)=>{
  console.log("1")
  resolve()
}).then(()=>{        
  console.log("2")
  new Promise((resolve,reject)=>{
    console.log("3")
    resolve()
  }).then(()=>{      
    console.log("4")
  }).then(()=>{       
    console.log("5")
  })
}).then(()=>{  
  console.log("6")
})

new Promise((resolve,reject)=>{
  console.log("7")
  resolve()
}).then(()=>{         
  console.log("8")
})

/*
	输出结果为:1 7 2 3 8 4 6 5 0
	1 7 为同步
*/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值