昨天看了一篇vue的教程,作者用async/ await来发送异步请求,从服务端获取数据,代码很简洁,同时async/await 已经被标准化,是时候学习一下了。

  先说一下async的用法,它作为一个关键字放到函数前面,用于表示函数是一个异步函数,因为async就是异步的意思, 异步函数也就意味着该函数的执行不会阻塞后面代码的执行。 写一个async 函数

  1. async function timeout() {
      return 'hello world';
    }

   语法很简单,就是在函数前面加上async 关键字,来表示它是异步的,那怎么调用呢?async 函数也是函数,平时我们怎么使用函数就怎么使用它,直接加括号调用就可以了,为了表示它没有阻塞它后面代码的执行,我们在async 函数调用之后加一句console.log;

  1. async function timeout() {
  2. return 'hello world'
  3. }
  4. timeout();
  5. console.log('虽然在后面,但是我先执行');

  打开浏览器控制台,我们看到了

  async 函数 timeout  调用了,但是没有任何输出,它不是应该返回 'hello world',  先不要着急, 看一看timeout()执行返回了什么? 把上面的 timeout() 语句改为console.log(timeout())

  1. async function timeout() {
  2. return 'hello world'
  3. }
  4. console.log(timeout());
  5. console.log('虽然在后面,但是我先执行');

  继续看控制台

  原来async 函数返回的是一个promise 对象,如果要获取到promise 返回值,我们应该用then 方法, 继续修改代码

  1. async function timeout() {
  2. return 'hello world'
  3. }
  4. timeout().then(result => {
  5. console.log(result);
  6. })
  7. console.log('虽然在后面,但是我先执行');

  看控制台

  我们获取到了"hello world',  同时timeout 的执行也没有阻塞后面代码的执行,和 我们刚才说的一致。

  这时,你可能注意到控制台中的Promise 有一个resolved,这是async 函数内部的实现原理。如果async 函数中有返回一个值 ,当调用该函数时,内部会调用Promise.solve() 方法把它转化成一个promise 对象作为返回,但如果timeout 函数内部抛出错误呢? 那么就会调用Promise.reject() 返回一个promise 对象, 这时修改一下timeout 函数

  1. async function timeout(flag) {
  2. if (flag) {
  3. return 'hello world'
  4. } else {
  5. throw 'my god, failure'
  6. }
  7. }
  8. console.log(timeout(true)) // 调用Promise.resolve() 返回promise 对象。
  9. console.log(timeout(false)); // 调用Promise.reject() 返回promise 对象。

  控制台如下:

  如果函数内部抛出错误, promise 对象有一个catch 方法进行捕获。

  1. timeout(false).catch(err => {
  2. console.log(err)
  3. })

  async 关键字差不多了,我们再来考虑await 关键字,await是等待的意思,那么它等待什么呢,它后面跟着什么呢?其实它后面可以放任何表达式,不过我们更多的是放一个返回promise 对象的表达式。注意await 关键字只能放到async 函数里面

  现在写一个函数,让它返回promise 对象,该函数的作用是2s 之后让数值乘以2

  1. // 2s 之后返回双倍的值
  2. function doubleAfter2seconds(num) {
  3. return new Promise((resolve, reject) => {
  4. setTimeout(() => {
  5. resolve(2 * num)
  6. }, 2000);
  7. } )
  8. }

  现在再写一个async 函数,从而可以使用await 关键字, await 后面放置的就是返回promise对象的一个表达式,所以它后面可以写上 doubleAfter2seconds 函数的调用

  1. async function testResult() {
  2. let result = await doubleAfter2seconds(30);
  3. console.log(result);
  4. }

  现在调用testResult 函数

  1. testResult();

  打开控制台,2s 之后,输出了60.

  现在我们看看代码的执行过程,调用testResult 函数,它里面遇到了await, await 表示等一下,代码就暂停到这里,不再向下执行了,它等什么呢?等后面的promise对象执行完毕,然后拿到promise resolve 的值并进行返回,返回值拿到之后,它继续向下执行。具体到 我们的代码, 遇到await 之后,代码就暂停执行了, 等待doubleAfter2seconds(30) 执行完毕,doubleAfter2seconds(30) 返回的promise 开始执行,2秒 之后,promise resolve 了, 并返回了值为60, 这时await 才拿到返回值60, 然后赋值给result, 暂停结束,代码才开始继续执行,执行 console.log语句。

  就这一个函数,我们可能看不出async/await 的作用,如果我们要计算3个数的值,然后把得到的值进行输出呢?

  1. async function testResult() {
  2. let first = await doubleAfter2seconds(30);
  3. let second = await doubleAfter2seconds(50);
  4. let third = await doubleAfter2seconds(30);
  5. console.log(first + second + third);
  6. }

  6秒后,控制台输出220, 我们可以看到,写异步代码就像写同步代码一样了,再也没有回调地域了。

  再写一个真实的例子,我原来做过一个小功能,话费充值,当用户输入电话号码后,先查找这个电话号码所在的省和市,然后再根据省和市,找到可能充值的面值,进行展示。

为了模拟一下后端接口,我们新建一个node 项目。 新建一个文件夹 async, 然后npm init -y 新建package.json文件,npm install express --save 安装后端依赖,再新建server.js 文件作为服务端代码, public文件夹作为静态文件的放置位置, 在public 文件夹里面放index.html 文件, 整个目录如下

  server.js 文件如下,建立最简单的web 服务器

  1. const express = require('express');
  2. const app = express();// express.static 提供静态文件,就是html, css, js 文件
  3. app.use(express.static('public'));
  4.  
  5. app.listen(3000, () => {
  6. console.log('server start');
  7. })

  再写index.html 文件,我在这里用了vue构建页面,用axios 发送ajax请求, 为了简单,用cdn 引入它们。 html部分很简单,一个输入框,让用户输入手机号,一个充值金额的展示区域, js部分,按照vue 的要求搭建了模版

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Async/await</title>
  6. <!-- CDN 引入vue 和 axios -->
  7. <script src="https://cdn.jsdelivr.net/npm/vue"></script>
  8. <script src="https://unpkg.com/axios/dist/axios.min.js"></script>
  9. </head>
  10. <body>
  11. <div id="app">
  12.  
  13. <!-- 输入框区域 -->
  14. <div style="height:50px">
  15. <input type="text" placeholder="请输入电话号码" v-model="phoneNum">
  16. <button @click="getFaceResult">确定</button>
  17. </div>
  18.  
  19. <!-- 充值面值 显示区域 -->
  20. <div>
  21. 充值面值:
  22. <span v-for="item in faceList" :key='item'>
  23. {{item}}
  24. </span>
  25. </div>
  26. </div>
  27.  
  28. <!-- js 代码区域 -->
  29. <script>
  30. new Vue({
  31. el: '#app',
  32. data: {
  33. phoneNum: '12345',
  34. faceList: ["20元", "30元", "50元"]
  35. },
  36. methods: {
  37. getFaceResult() {
  38.  
  39. }
  40. }
  41. })
  42. </script>
  43. </body>
  44. </html>

  为了得到用户输入的手机号,给input 输入框添加v-model指令,绑定phoneNum变量。展示区域则是 绑定到faceList 数组,v-for 指令进行展示, 这时命令行nodemon server 启动服务器,如果你没有安装nodemon, 可以npm install -g nodemon 安装它。启动成功后,在浏览器中输入 http://localhost:3000, 可以看到页面如下, 展示正确

  现在我们来动态获取充值面值。当点击确定按钮时, 我们首先要根据手机号得到省和市,所以写一个方法来发送请求获取省和市,方法命名为getLocation, 接受一个参数phoneNum , 后台接口名为phoneLocation,当获取到城市位置以后,我们再发送请求获取充值面值,所以还要再写一个方法getFaceList, 它接受两个参数, province 和city, 后台接口为faceList,在methods 下面添加这两个方法getLocation, getFaceList

  1. methods: {
  2. //获取到城市信息
  3. getLocation(phoneNum) {
  4. return axios.post('phoneLocation', {
  5. phoneNum
  6. })
  7. },
  8. // 获取面值
  9. getFaceList(province, city) {
  10. return axios.post('/faceList', {
  11. province,
  12. city
  13. })
  14. },
  15. // 点击确定按钮时,获取面值列表
  16. getFaceResult () {
  17.  
  18. }
  19. }

  现在再把两个后台接口写好,为了演示,写的非常简单,没有进行任何的验证,只是返回前端所需要的数据。Express 写这种简单的接口还是非常方便的,在app.use 和app.listen 之间添加如下代码

  1. // 电话号码返回省和市,为了模拟延迟,使用了setTimeout
  2. app.post('/phoneLocation', (req, res) => {
  3. setTimeout(() => {
  4. res.json({
  5. success: true,
  6. obj: {
  7. province: '广东',
  8. city: '深圳'
  9. }
  10. })
  11. }, 1000);
  12. })
  13.  
  14. // 返回面值列表
  15. app.post('/faceList', (req, res) => {
  16. setTimeout(() => {
  17. res.json(
  18. {
  19. success: true,
  20. obj:['20元', '30元', '50元']
  21. }
  22.  
  23. )
  24. }, 1000);
  25. })

  最后是前端页面中的click 事件的getFaceResult, 由于axios 返回的是promise 对象,我们使用then 的链式写法,先调用getLocation方法,在其then方法中获取省和市,然后再在里面调用getFaceList,再在getFaceList 的then方法获取面值列表,

  1. // 点击确定按钮时,获取面值列表
  2. getFaceResult () {
  3. this.getLocation(this.phoneNum)
  4. .then(res => {
  5. if (res.status === 200 && res.data.success) {
  6. let province = res.data.obj.province;
  7. let city = res.data.obj.city;
  8.  
  9. this.getFaceList(province, city)
  10. .then(res => {
  11. if(res.status === 200 && res.data.success) {
  12. this.faceList = res.data.obj
  13. }
  14. })
  15. }
  16. })
  17. .catch(err => {
  18. console.log(err)
  19. })
  20. }

  现在点击确定按钮,可以看到页面中输出了 从后台返回的面值列表。这时你看到了then 的链式写法,有一点回调地域的感觉。现在我们在有async/ await 来改造一下。

首先把 getFaceResult 转化成一个async 函数,就是在其前面加async, 因为它的调用方法和普通函数的调用方法是一致,所以没有什么问题。然后就把 getLocation 和

getFaceList 放到await 后面,等待执行, getFaceResult  函数修改如下

  1. // 点击确定按钮时,获取面值列表
  2. async getFaceResult () {
  3. let location = await this.getLocation(this.phoneNum);
  4. if (location.data.success) {
  5. let province = location.data.obj.province;
  6. let city = location.data.obj.city;
  7. let result = await this.getFaceList(province, city);
  8. if (result.data.success) {
  9. this.faceList = result.data.obj;
  10. }
  11. }
  12. }

  现在代码的书写方式,就像写同步代码一样,没有回调的感觉,非常舒服。

  现在就还差一点需要说明,那就是怎么处理异常,如果请求发生异常,怎么处理? 它用的是try/catch 来捕获异常,把await 放到 try 中进行执行,如有异常,就使用catch 进行处理。

  1. async getFaceResult () {
  2. try {
  3. let location = await this.getLocation(this.phoneNum);
  4. if (location.data.success) {
  5. let province = location.data.obj.province;
  6. let city = location.data.obj.city;
  7. let result = await this.getFaceList(province, city);
  8. if (result.data.success) {
  9. this.faceList = result.data.obj;
  10. }
  11. }
  12. } catch(err) {
  13. console.log(err);
  14. }
  15. }

  现在把服务器停掉,可以看到控制台中输出net Erorr,整个程序正常运行。

ES6中async和await说明和用法的更多相关文章

  1. ES6中async与await的使用方法

    promise的使用方法 promise简介 是异步编程的一种解决方案.从语法上说,Promise 是一个对象,从它可以获取异步操作的消息.解决回调函数嵌套过多的情况 const promise =n ...

  2. C# 中 async 和 await 的基本使用

    C# 中 async 和 await 的基本使用 前言 经常在 C# 的代码中看到以 Async 结尾的方法,大概知道意为异步方法,但不知道怎么使用,也不知道如何定义. 对于"同步" ...

  3. JavaScript中async和await的使用以及队列问题

    宏任务和微任务的队列入门知识,可以参考之前的文章: JavaScript的事件循环机制 宏任务和微任务在前端面试中,被经常提及到,包括口头和笔试题 async && await概念 a ...

  4. 反爬虫:利用ASP.NET MVC的Filter和缓存(入坑出坑) C#中缓存的使用 C#操作redis WPF 控件库——可拖动选项卡的TabControl 【Bootstrap系列】详解Bootstrap-table AutoFac event 和delegate的分别 常见的异步方式async 和 await C# Task用法 c#源码的执行过程

    反爬虫:利用ASP.NET MVC的Filter和缓存(入坑出坑)   背景介绍: 为了平衡社区成员的贡献和索取,一起帮引入了帮帮币.当用户积分(帮帮点)达到一定数额之后,就会“掉落”一定数量的“帮帮 ...

  5. C#中async和await用法

    .net 4.5中新增了async和await这一对用于异步编程的关键字. async放在方法中存在await代码的方法中,await放在调用返回Task的方法前. class Class1 { pr ...

  6. ES6之async与await

    · async - await 是 Promise 和 Generator 的语法糖,目的只是为了让我们书写代码时更加流畅,增强代码的可读性. · async - await 是建立在Promise机 ...

  7. ES6中的Array.from()函数的用法

    ES6为Array增加了from函数用来将其他对象转换成数组. 当然,其他对象也是有要求,也不是所有的,可以将两种对象转换成数组. 1.部署了Iterator(迭代器)接口的对象,比如:Set,Map ...

  8. Async和await关键字的用法

    async & await 的前世今生(Updated) 1. 方法打上Async关键字, 就可以使用await调用别的Async方法了 2. 记得在需要异步执行的方法里面调用await或者n ...

  9. js中回调函数,promise 以及 async/await 的对比用法 对比!!!

    在编程项目中,我们常需要用到回调的做法来实现部分功能,那么在js中我们有哪些方法来实现回调的? 方法1:回调函数 首先要定义这个函数,然后才能利用回调函数来调用! login: function (f ...

随机推荐

  1. select 语句中 if 的用法

    IF( expr1 , expr2 , expr3 ) expr1 的值为 TRUE,则返回值为 expr2 expr1 的值为FALSE,则返回值为 expr3 例: ,); ,); ", ...

  2. day 42 01--CSS的引入方式及CSS选择器

    01--CSS的引入方式及CSS选择器   本节目录 一 CSS介绍 二 行内样式 三 内接样式 四 外接样式 五 CSS的选择器 六 CSS的高级选择器 七 CSS的属性选择器 八 CSS的伪类选择 ...

  3. idea怎么打war包

    1.上方导航栏粥找到 Buid——> Bild Artifacts... 2.弹出框中选择 3.war包打好啦,一般放在编译的 target目录下

  4. sql草稿

    参考:MySQL 内连接.外连接.左连接.右连接.全连接 SELECT count(*) FROM `t_product_base` select m_name from t_medicinal_in ...

  5. Leetcode152. Maximum Product Subarray乘积的最大子序列

    给定一个整数数组 nums ,找出一个序列中乘积最大的连续子序列(该序列至少包含一个数). 示例 1: 输入: [2,3,-2,4] 输出: 6 解释: 子数组 [2,3] 有最大乘积 6. 示例 2 ...

  6. MySQL的高可用实现方案之mysql-mmm

    一.环境简述 1.工作逻辑图 2.MySQL-MMM优缺点 优点:高可用性,扩展性好,出现故障自动切换,对于主主同步,在同一时间只提供一台数据库写操作,保证的数据的一致性. 缺点:Monitor节点是 ...

  7. Django项目:CRM(客户关系管理系统)--66--56PerfectCRM实现CRM客户报名缴费链接

    # kingadmin.py # ————————04PerfectCRM实现King_admin注册功能———————— from crm import models #print("ki ...

  8. 史上最直接小白式的Sourcetree的分支创建与合并

    一.Sourcetree简单介绍通过Git可以进行对项目的版本管理,但是如果直接使用Git的软件会比较麻烦,因为是通过一条一条命令进行操作的. Sourcetree则可以与Git结合,提供图形界面,使 ...

  9. 19.10.14-Q

    小$P$的咕事 总结: 还行,就是$T1$写的慢了,$T2,T3$暴力有点锅 T1 小模拟. 打就是了. 可以小小的手玩一下. (考试的时候某同志人肉对拍了$20min$)=.= 418 ms 360 ...

  10. Touching segments(数据结构)

    题目链接 Problem Statement Your Maths professor is a very nice guy, but he sometimes comes up with not s ...