用 async/await 来处理异步

前言


最近在通过jest/pupppteer写测试用例,async/ await来发送异步请求,模拟用户才做,async/await 已经被标准化,是时候学习一下了。

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

    async function timeout() {
    return 'hello world';
    }
语法很简单,就是在函数前面加上async 关键字,来表示它是异步的,那怎么调用呢?async 函数也是函数,平时我们怎么使用函数就怎么使用它,直接加括号调用就可以了,为了表示它没有阻塞它后面代码的执行,我们在async 函数调用之后加一句console.log
    async function timeout() {
        return 'hello world'
    }
    timeout();
    console.log('虽然在后面,但是我先执行');
打开浏览器控制台,我们看到了

console.png

async 函数返回的是一个promise 对象,如果要获取到promise 返回值,我们应该用then 方法
    async function timeout() {
        return 'hello world'
    }
    timeout().then(result => {
    console.log(result);
    })
    console.log('虽然在后面,但是我先执行');
控制台如下:

console1.png

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

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

    async function timeout(flag) {
        if (flag) {
            return 'hello world'
        } else {
            throw 'my god, failure'
        }
    }
    console.log(timeout(true))  // 调用Promise.resolve() 返回promise 对象。
    console.log(timeout(false)); // 调用Promise.reject() 返回promise 对象。
控制台如下:

console2.png

如果函数内部抛出错误, promise 对象有一个catch 方法进行捕获。
    timeout(false).catch(err => {
        console.log(err)
    })
async 关键字差不多了,我们再来考虑await 关键字,await是等待的意思,那么它等待什么呢,它后面跟着什么呢?其实它后面可以放任何表达式,不过我们更多的是放一个返回promise 对象的表达式。注意await 关键字只能放到async 函数里面

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

// 2s 之后返回双倍的值
    function doubleAfter2seconds(num) {
        return new Promise((resolve, reject) => {
            setTimeout(() => {
                resolve(2 * num)
            }, 2000);
        } )
    }
现在再写一个async 函数,从而可以使用await 关键字, await 后面放置的就是返回promise对象的一个表达式,所以它后面可以写上 doubleAfter2seconds 函数的调用
    async function testResult() {
        let result = await doubleAfter2seconds(30);
        console.log(result);
    }
    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个数的值,然后把得到的值进行输出呢?

    async function testResult() {
        let first = await doubleAfter2seconds(30);
        let second = await doubleAfter2seconds(50);
        let third = await doubleAfter2seconds(30);
        console.log(first + second + third);
    }
打开控制台,6秒后,控制台输出220, 我们可以看到,写异步代码就像写同步代码一样了,再也没有回调地域了。

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

为了模拟一下后端接口,我们新建一个node 项目。

新建一个文件夹 async, 然后新建package.json文件
npm init -y
安装express
npm install express --save
新建server.js 文件作为服务端代码
const express = require('express');
const app = express();// express.static 提供静态文件,就是html, css, js 文件
app.use(express.static('public'));

app.listen(3000, () => {
    console.log('server start');
})
新建public文件夹作为静态文件的放置位置, 在public 文件夹里面放index.html 文件

整个目录如下

用 async/await 来处理异步_第1张图片

再写index.html 文件,这里用了vue构建页面,通过 axios 发送ajax请求, 用cdn引入

html部分,一个输入框,让用户输入手机号,一个充值金额的展示区域

js部分,按照vue 的要求搭建模版


    
    
        
        Async/await
        
        
        
    
    
        
充值面值: {{item}}
为了得到用户输入的手机号,给input 输入框添加v-model指令,绑定phoneNum变量,展示区域则是 绑定到faceList 数组,v-for 指令进行展示。命令行输入node server.js 启动服务器。启动成功后,在浏览器中输入 http://localhost:3000,
可以看到页面如下

display.png

现在我们来动态获取充值面值。当点击确定按钮时, 我们首先要根据手机号得到省和市,所以写一个方法来发送请求获取省和市,方法命名为getLocation, 接受一个参数phoneNum , 后台接口名为phoneLocation,当获取到城市位置以后,我们再发送请求获取充值面值,所以还要再写一个方法getFaceList, 它接受两个参数, province 和city, 后台接口为faceList,在methods 下面添加这两个方法getLocation, getFaceList
        methods: {
            //获取到城市信息
            getLocation(phoneNum) {
               return axios.post('phoneLocation', {
                    phoneNum
                })
            },
            // 获取面值
            getFaceList(province, city) {
                return axios.post('/faceList', {
                    province,
                    city
                })
            },
            // 点击确定按钮时,获取面值列表
            getFaceResult () {
               
            }
        }
现在再把两个后台接口写好,为了演示,写的非常简单,没有进行任何的验证,只是返回前端所需要的数据。Express 写这种简单的接口还是非常方便的,在app.use 和app.listen 之间添加如下代码
// 电话号码返回省和市,为了模拟延迟,使用了setTimeout
app.post('/phoneLocation', (req, res) => {
    setTimeout(() => {
        res.json({
            success: true,
            obj: {
                province: '广东',
                city: '深圳'
            }
        })
    }, 1000);
})

// 返回面值列表
app.post('/faceList', (req, res) => {
    setTimeout(() => {
        res.json(
            {
                success: true,
                obj:['20元', '30元', '50元']
            }
            
        )
    }, 1000);
})

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

      // 点击确定按钮时,获取面值列表
      getFaceResult () {
          this.getLocation(this.phoneNum)
              .then(res => {
                   if (res.status === 200 && res.data.success) {
                       let province = res.data.obj.province;
                       let city = res.data.obj.city;
                       this.getFaceList(province, city)
                           .then(res => {
                               if(res.status === 200 && res.data.success) {
                                   this.faceList = res.data.obj
                               }
                           })
                       }
                   }).catch(err => {
                        console.log(err)
              })
      }
现在点击确定按钮,可以看到页面中输出了 从后台返回的面值列表。这时你看到了then 的链式写法,有一点回调地域的感觉。现在我们在有async/ await 来改造一下。

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

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

     // 点击确定按钮时,获取面值列表
     async getFaceResult () {
           let location = await this.getLocation(this.phoneNum);
           if (location.data.success) {
               let province = location.data.obj.province;
               let city = location.data.obj.city;
               let result = await this.getFaceList(province, city);
               if (result.data.success) {
                   this.faceList = result.data.obj;
               }
           }
     }
现在代码的书写方式,就像写同步代码一样,没有回调的感觉,非常舒服。

处理异常,如果请求发生异常,怎么处理? 它用的是try/catch 来捕获异常,把await 放到 try 中进行执行,如有异常,就使用catch 进行处理。

     async getFaceResult () {
            try {
                let location = await this.getLocation(this.phoneNum);
                if (location.data.success) {
                    let province = location.data.obj.province;
                    let city = location.data.obj.city;
                    let result = await this.getFaceList(province, city);
                    if (result.data.success) {
                        this.faceList = result.data.obj;
                    }
                }
            } catch(err) {
                console.log(err);
         }
     }
现在把服务器停掉,可以看到控制台中输出net Erorr,整个程序正常运行。

你可能感兴趣的:(javascript,node.js,html5,前端)