02 JavaStrip 任务队列与事件循环机制

上一节我们讲了js是单线程,但在数据请求上会有一些问题,通常我们会用ajax异步的去请求数据,也能保证我们项目正常的运行,那js中异步模式的原理有是怎么样的呢?

js虽然是单线程,但也提供了一个任务队列,利用任务队列我们可以模拟一个多线程,里面放的任务都是一些异步的操作。我们在函数的章节讲过,决定函数的执行顺序是又call stask(调用栈),可js不仅仅有函数,还有其他的模式,而其他模式是又称事件循环

任务队列(消息队列)

1、是一个有序的任务集合列表(放的都是异步任务),在js中任务队列可以有多个;

任务队列分类:

task(macrotask)宏任务队列 microtask( 微任务队列)
DOM操作 Promise
用户交互(事件) process.nextTick(nodejs)
网络任务(ajax) MutationObserver(H5里监听DOM节点变化)
history traversal任务(h5当中的历史操作)
定时器

2、当代码里有异步操作的时候,就会把异步操作分发到对应的任务对列里;

它们的执行顺序也是不同的,先走微任务,后走宏任务。那任务队列的代码是怎么放进去的?其实是有一个任务源

3、每一个任务都会有任务源;

任务源是用来分发任务的,根据任务的不同来决定是放在宏任务还是微任务

4、一个队列里可以放多个任务,任务的读取方式为先进先出,后进后出;

栗子一:

setTimeout(()=>{
  console.log('老沙最慢');
},100);
setTimeout(()=>{
  console.log('八戒第二');
},0);
console.log('猴哥最快');

上面的代码是怎么执行的呢?
当代码运行的时候,先创建一个globalEC,代码继续运行的时候发现一个定时器,定时器就是一个任务源,它会把定位器代码放入宏任务中,但定时器有一个100毫秒的时间,意思是100毫秒后才把定期器代码放到宏任务中,接着往下走,又发现一个定时器,而定时的时间为0毫秒,所以会先把‘八戒第二’这个任务放到宏任务中,此时所有的代码依然没有执行,代码要执行还得需放入call stack中执行,代码继续往下走,发现了“console.log('猴哥最快')”,而“console.log('猴哥最快');”并不是宏任务或者微任务,则直接放到call stack中,然后执行,一旦执行完成,就从call stack中弹出,然后开始执行微任务,但是此时,微任务什么也没有,就开始执行宏任务,先把定时器中“ 八戒第二” 先放到call stack中执行,执行完成后弹出,最后在把“ 老沙最慢”的定时器放入call stack 中执行。所以会依次打印 “ 猴哥最快 八戒第二 老沙最慢”

事件循环 Event Loop

1、它是HTML规范中定义的或者是DOM中定义的,并不是ECMAScript定义的;

    参考(打不开的话需翻墙)
    1、https://www.w3.org/TR/html5/webappapis.html#event-loops
    2、https://html.spec.whatwg.org/#event-loops

2、它包括两种,一种在浏览器上下文里,一种在web workers里(HTML5);
3、它是一种分配模式,把任务队列里的任务分配给全局执行上下文进行执行;
4、执行顺序;
    1、先把微任务队列里的任务拿出来交给全局执行上下文执行;   
    2、再把宏任务队列里的任务拿出来交给全局执行上下文执行;
    3、不断的循环;

注意:

1、任务队列(异步代码)里的代码最终还是要放到全局执行上下文(同步的代码)里去执行;
2、当全局执行上下文里的代码都执行完了才去执行任务队列里的代码;
3、宏任务与微任务的区别是:宏任务会再下一次的Event Loop里执行,微任务会在本次Event Loop里执行;

栗子二:

console.log('全局的');
setTimeout(()=>{
        console.log('定时器的');
},0);           
new Promise((resolve,reject)=>{
    console.log('Promise的');
    resolve();
}).then(()=>{
    console.log('then的');
});

以上代码运行的时候,首先发现一个“console.log('全局的');”,会首先放入到call stack里执行,然后继续运行的时候,发现一个定时器,会把定时器放入到宏任务当中,继续运行的时候,又发现了一个new Promise,在new Promise的回调函数中的“ console.log('定时器的');”会放入全局执行栈中立即执行,resolve()也会放到全局执行栈中,而Promise的then代码放入到微任务当中,先从微任务中把“console.log('then的');”发入call stack中执行,执行完成之后再弹出去,此时微任务中已经没有代码可以执行了,微任务 事件循环已经完成,此时一遍任务循环一节完成了,然后再进行第二次任务循环,开始执行宏任务 的代码,所以打印的结果依次是“全局的、Promise的、then的、定时器的”。

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

推荐阅读更多精彩内容