JavaScript宏观任务和微观任务

JavaScript代码在执行的时候,可以说就是拿一段代码给到JavaScript引擎并去执行,此外还可能会提供额外的API给到JavaScript引擎。

在ES3 或者 更早的版本中,JavaScript并无异步操作,所以代码给到JavaScript引擎,它就直接顺次的执行,这个任务是宿主发起的任务我们可以称之为宏观任务(macrotask)。

在ES5 或者 之后的版本,JavaScript出现了Promise,这就不需要浏览器的安排,引擎自己也可以发起任务,这个任务就叫做微观任务(microtask)

image.png

Promise

MDN解释:Promise 对象用于表示一个异步操作的最终状态(完成或失败),以及该异步操作的结果值。

基本用法:

functionsleep(duration){returnnewPromise(function(resolve,reject){setTimeout(resolve,duration);})}sleep(1000).then(()=>console.log("then")).catch(()=>console.log('catch')).finally(()=>console.log('finally'));

Promise.prototype.catch(onRejected)

添加一个拒绝(rejection) 回调到当前 promise, 返回一个新的promise。当这个回调函数被调用,新 promise 将以它的返回值来resolve,否则如果当前promise 进入fulfilled状态,则以当前promise的完成结果作为新promise的完成结果.

Promise.prototype.then(onFulfilled, onRejected)

添加解决(fulfillment)和拒绝(rejection)回调到当前 promise, 返回一个新的 promise, 将以回调的返回值来resolve.

Promise.prototype.finally(onFinally)

添加一个事件处理回调于当前promise对象,并且在原promise对象解析完毕后,返回一个新的promise对象。回调会在当前promise运行完毕后被调用,无论当前promise的状态是完成(fulfilled)还是失败(rejected)

functionsleep(duration){returnnewPromise(function(resolve,reject){console.log("b");setTimeout(resolve,duration);})}console.log("a");sleep(5000).then(()=>console.log("c"));

setTimeout 是属于宿主环境的api,属于宏观任务,所以可以分析为两个宏观任务,但是setTimeout中带有一个微观任务。所以执行结果为a b c

async/await

当调用一个 async 函数时,会返回一个 Promise 对象。当这个 async 函数返回一个值时,Promise 的 resolve 方法会负责传递这个值;当 async 函数抛出异常时,Promise 的 reject 方法也会传递这个异常值。

async 函数中可能会有 await 表达式,这会使 async 函数暂停执行,等待 Promise 的结果出来,然后恢复async函数的执行并返回解析值(resolved)。

注意, await 关键字仅仅在 async function中有效。如果在 async function函数体外使用 await ,你只会得到一个语法错误(对象代表尝试解析语法上不合法的代码的错误。)。

varresolveAfter2Seconds=function(){console.log("starting slow promise");returnnewPromise(resolve=>{setTimeout(function(){resolve("slow");console.log("slow promise is done");},2000);});};varresolveAfter1Second=function(){console.log("starting fast promise");returnnewPromise(resolve=>{setTimeout(function(){resolve("fast");console.log("fast promise is done");},1000);});};varsequentialStart=asyncfunction(){console.log('==SEQUENTIAL START==');// 1. Execution gets here almost instantlyconstslow=awaitresolveAfter2Seconds();console.log(slow);// 2. this runs 2 seconds after 1.constfast=awaitresolveAfter1Second();console.log(fast);// 3. this runs 3 seconds after 1.}varconcurrentStart=asyncfunction(){console.log('==CONCURRENT START with await==');constslow=resolveAfter2Seconds();// starts timer immediatelyconstfast=resolveAfter1Second();// starts timer immediately// 1. Execution gets here almost instantlyconsole.log(awaitslow);// 2. this runs 2 seconds after 1.console.log(awaitfast);// 3. this runs 2 seconds after 1., immediately after 2., since fast is already resolved}varconcurrentPromise=function(){console.log('==CONCURRENT START with Promise.all==');returnPromise.all([resolveAfter2Seconds(),resolveAfter1Second()]).then((messages)=>{console.log(messages[0]);// slowconsole.log(messages[1]);// fast});}varparallel=asyncfunction(){console.log('==PARALLEL with await Promise.all==');// Start 2 "jobs" in parallel and wait for both of them to completeawaitPromise.all([(async()=>console.log(awaitresolveAfter2Seconds()))(),(async()=>console.log(awaitresolveAfter1Second()))()]);}// This function does not handle errors. See warning below!varparallelPromise=function(){console.log('==PARALLEL with Promise.then==');resolveAfter2Seconds().then((message)=>console.log(message));resolveAfter1Second().then((message)=>console.log(message));}sequentialStart();// after 2 seconds, logs "slow", then after 1 more second, "fast"// wait above to finishsetTimeout(concurrentStart,4000);// after 2 seconds, logs "slow" and then "fast"// wait againsetTimeout(concurrentPromise,7000);// same as concurrentStart// wait againsetTimeout(parallel,10000);// truly parallel: after 1 second, logs "fast", then after 1 more second, "slow"// wait againsetTimeout(parallelPromise,13000);// same as parallel

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