JavaScript异步——callback、promise、async/await

背景

JavaScript是单线程工作,这意味着两段脚本不能同时运行,而且必须一个接一个的运行。

其实JavaScript的单线程与它的用途有很大的关系,JavaScript作为浏览器脚本语言,主要实现与用户的交互。利用JavaScript可以对DOM做各种各样的操作。若JavaScript是多线程的话,一个线程在一个DOM节点中增加内容,另一个线程要删除这个DOM节点。那么这个DOM节点就很纠结,这个DOM节点到底要增加内容还是要删除呢?因此JavaScript是单线程的。

同步任务与异步任务

由于JavaScript的单线程特性,因此同一时间只能处理同一个任务,所有任务都需要排队,前一个任务执行完,才可以执行下一个任务。

但是如果前一个任务的执行时间很长,比如说是文件的读取操作或Ajax操作,后一个任务就不得不等待。就比如是Ajax,当用户向后台获取大量数据时,必须等到所有的数据都获取完才能进行下一步的操作,用户就只能等待,严重影响用户体验。

在JavaScript的设计之初就考虑到了这个问题。主线程可以完全不管设备I/O这种耗时的任务,会挂起处于等待任务;先运行排在后面的任务。等到挂起的任务返回了接轨后,再对挂起的任务进行后续处理。因此任务可以分为同步任务和异步任务。

  • 同步任务:同步任务指在主线程上排队的任务,只有前一个任务执行完毕,才能继续执行下一个任务。例如WEB页面的渲染过程就是一个同步任务。
  • 异步任务:异步任务是指不进入主线程,而进入任务队列的任务。只有任务队列通知主线程,某个异步任务可以执行了。该任务才进入主线程执行。例如图片、音乐资源的加载都是一个异步任务。

具体来说JavaScript任务的执行机制如下:

1.  所有同步任务都在主线程上执行,形成一个执行栈(execution context stack)。
2.  主线程之外,还存在一个“任务队列”(task queue),只要异步任务有了运行结果,就在“任务队列”中放置一个事件。
3.  一旦“执行栈”中的所有同步任务执行完毕,系统就会读取“任务队列”,看看里边有哪些事件。哪些对应的异步任务,于是结束等待状态,进入“执行栈”开始执行。
4.  主线程不断重复上边的第三步。

JavaScript的主线程和任务队列示意图:

3310_1.png

JavaScript异步编程

在JavaScript中通常使用回调函数、Promise以及async/await的方式实现异步。

回调函数

回调函数是实现异步编程最简单的方式。具体的做法是定义一个函数,将这个函数绑定到事件上,当触发事件后会自动调用这个函数,不需要主动去调用这个函数,称这个函数为回调函数。
例如:

var req = new XMLHttpReauest()
req.open("GET", url)
req.send(null)
req.onreadystatechange=function() {}

onreadystatechange函数上绑定的这个函数就称为是回调函数。

回调具体可以分为具名回调、匿名回调和回调地狱

具名回调

function getUserInfo(fn) {
    fn.call(undefined, 'name: xxx')
}

function userinfo(info) {
    console.log(info)
}

getUserInfo.call(undefined, userinfo)

// name: xxx

匿名回调

function getUserInfo(fn) {
    fn.call(undefined, 'name: xxx')
}

getUserInfo.call(undefined, function(info) {
    console.log(info)
})

多层嵌套的匿名回调(回调地狱)

function getUserInfo(fn) {
    fn.call(undefined, 'name: xxx')
}

function saveUserInfo(fn) {
    fn.call(undefined, 'name: xxx')
}

function getOtherUserInfo(fn) {
    fn.call(undefined, 'name: xxx')
}

getUserInfo.call(undefined, function(info) {
    console.log(info)
    saveUserInfo.call(undefined, function() {
        getOtherUserInfo.call(undefined, function() {
            saveUserInfo.call(undefined, function() {
                ......
            })
        })
    })
})

向上面这种多层匿名回调嵌套就很难读懂和维护,这种代码就称为回调地狱。

回调函数的优点是写法简单,但是容易出现回调地狱。

Promise

Promise对象是CommonJS定义的一种规范,目的为异步编程提供统一的接口。

Promise包括以下几个规范:

  • 一个promise可能有三种状态:等待( pending )、已完成( fulfilled )和已拒绝( rejected )
  • 一个promise的状态只可能从等待转到完成拒绝,不能逆向转换,同时完成拒绝不能相互转换。
  • promise必须实现一个then方法,而且then方法必须返回一个promise,同一个promise的then可以调用多次,并且回调的执行顺序跟它们被定义时的顺序一致。
  • then方法接受两个参数,第一个参数是成功时的回调,在promise由等待转换为完成时调用;另一个参数是失败时的回调,在promise由等待转换为拒绝时调用。同时then可以接受另一个promise传入,也接受一个类then的对象或方法。
function wait(time) {
    return new Promise(function(resolve, reject) {
        setTimeout(resolve, time)
    })
}

wait(1000).then(function() {
    console.log(1)
})

async / await

从字面是理解async异步的意思,而await是等待的意思。所以async用于声明一个异步function,而await用于等待一个异步任务执行完成的结果。

其中await只能出现在async函数中,async函数的返回值是一个promise对象。

function test() {
    return new Promise(reslove => {
        setTimeout(() => reslove("test"), 2000)
    })
}

async function test2() {
    const result = await test()
    console.log(result)
}

test2()
console.log('end')

async的作用

通常情况下使用async命令是因为函数内部有await命令,因为await命令只能出现在async函数里面,否则会报语法,这就是为什么async/await成对出现的原因,但是如果对一个普通函数单独加个async会是什么结果呢?来看个例子:

async function test () {
    let a = 2
    return a
}

const res = test()
console.log(res)
3312_1.png

可以看到async函数的返回是一个promise对象。如果函数有返回值,async会把这个返回值通过promise.resole()封装成promise对象。通过then就可以将这个返回值取出来。

res.then(a => {
    console.log(a)      // 2
})

在没有await的情况下async函数会立即执行,并返回一个promise,那么加上await会有什么变化呢?

await的作用

一般情况下await命令后面接的是一个promise对象,等待promise对象状态发生变化,得到返回值,但是也可以接任意表达式的返回结果,例如:

function a () {
    return 'a'
}
async function b () {
    return 'b'
}

const c = await a()
const d = await b()
console.log(c, d)

可以看到await后面不管接什么表达式,都可以等到结果的返回。当等到的不是promise对象时,就将等到结果返回,当等到的是一个promise对象时,会阻塞后面的代码,等待promiset对象状态变化,得到对应的值作为await等待的结果,这里的阻塞是指async内部的阻塞,async函数的调用不会阻塞。

解决了什么问题

promise对象已经解决了回调地狱的问题,那么为什么还要async/await呢?看下面一段代码:

function login () {
    return new Promise(resolve => {
        resolve('aaa')
    })
}

function getUserInfo (token) {
    return new Promise(resolve => {
        if (token) {
            resolve({
                isVip: true
            })
        }
    })
}

function getVipGoods (userInfo) {
    return new Promise(resolve => {
        if (userInfo.isVip) {
            resolve({
                id: 'xxx',
                price: 'xxx'
            })
        }
    })
}

function showVipGoods (vipGoods) {
    console.log(vipGoods.id + '----' + vipGoods.price)
}

login()
    .then(token => getUserInfo(token))
    .then(userInfo => getVipGoods(userInfo))
    .then(vipGoods => showVipGoods(vipGoods))

上面的例子中,每一个promise都相对于是一个异步的网络请求,通常一个业务流对应了对个网络请求,上面的例子描述了每个网络请求都依赖前一个请求的结果的场景,下面采用async/awite重写。

async function call() {
    const token = await login()
    const userInfo = await getUserInfo(token)
    const vipGoods = await getVipGoods(userInfo)
    showVipGoods(vipGoods)
}

call()

相比于promise/then语法结构,使用async/await的调用更加清晰,和同步代码一样。

带来的问题

使用async/await会因为同步执行造成时间的积累,导致程序变慢。本质上async/await将并发执行的任务变为了继发。

在多个任务不关心执行顺序的情况下,继发会浪费很多的执行时间。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容