什么是同步?
在单线程的情况下,从上往下按顺序执行就是同步.
什么是异步?
通俗点解释很简单:不是同步的就是异步.也就是说他不是按顺序执行的.
那如何更深刻得去理解.
1)浏览器的渲染进程中JS解析线程就一个,所有的js代码都是他进行解析.
2)浏览器的渲染进程是有多个子线程的.这个我在<前端浅谈-浏览器工作进程>里有讲到过.这是能造成异步的条件之一.不同步了,就意味着有其他线程参加了.
3)同步会造成阻塞.当一个任务出现问题之后,下面的所有代码都会被阻塞掉.这对js来说是致命的.而类似http请求这种任务对于js来说是未知的,我不能冒险让它影响下面的所有代码.
4)异步问题一定是出现在回调函数上
所以,当一个任务的结果对于js来说会造成阻塞或者阻塞的可能性较大时,js解析线程会把它先交给其他线程,最后再去解析和执行回调函数.这一套操作叫做异步.而这些任务被称为异步任务.
宏任务和微任务
宏任务:宿主环境的函数或者方法 .有script(整体代码), setTimeout, setInterval, setImmediate, I/O, UI rendering
微任务:js的函数或者方法,有Promises, Object.observe, MutationObserver
关于宿主环境,我的<前端浅谈----宿主环境>里面有说过.在异步任务中,也会有先后顺序.具体的的顺序就要涉及到事件循环.
事件循环
所有的事件会在事件线程中,然后通过事件循环机制进行控制.它的最简单机制就是,先同再异,先微再宏.
比如:
console.log('同步1') setTimeout(()=>{ console.log('setTimeout1')
},0) //宏任务 new Promise((res)=>{
res() console.log('new Promise') //同步 }).then(()=>{ console.log('Promise.then') //微任务 })
console.log('同步2') //同步
结果
可以看到,基本就是按照这个顺序来执行的.其中有个注意点,这里的new Promise是同步的,但是promise.then()或者catch()这些才是异步.
以上这个流程只是一次流程,并没有把循环给体现出来,我们再看看
console.log('同步1') //同步 1 setTimeout(()=>{ //宏任务 console.log('setTimeout1') //宏任务中的同步 5 new Promise((res)=>{
res() console.log('new Promise1') //宏任务中的同步 6 }).then(()=>{ //宏任务中的微任务 8 console.log('Promise.then1') }) setTimeout(()=>{ //宏任务中的宏任务 9 console.log('setTimeout2') }) console.log('setTimeout同步') //宏任务中的同步 7 },0) new Promise((res)=>{ //同步 2 console.log('new Promise')
res() }).then(()=>{ //微任务 4 console.log('Promise.then') }) console.log('同步2') //同步 3
结果:
所谓的循环就是按照上面的流程递归执行,每进入到一个作用域就去执行来一遍,一直到最终把所有的事件执行完成.
这里还有个疑问,同是宏任务或者微任务,它们的执行顺序咋样?
setTimeout(()=>{ console.log('22') },1000) setInterval(() => { console.log('11') },1000)
其结果是22,11.反复尝试之后得到的结果是.对于同时间的延迟的事件,先写的先执行,不同时间的时间短的先执行.
如果是非计时器的函数是怎样的呢?
首先,事件执行的顺序跟推进队列的顺序有关.同作用域的情况下, process.nextTick有单独的事件列表,它的优先级最高.然后是微任务表,然后是宏任务列表.再循环.
js代码执行顺序(包含事件循环)
渲染进程包含以下线程
1.GUI线程(主要负责解析HTML、CSS和渲染页面) 2.JS引擎线程(负责解析和执行JS代码) 3.事件线程(控制事件循环) 4.定时器线程(处理定时器相关逻辑) 5.异步请求线程(发起Ajax时会生成该线程)
执行顺序
1. GUI线程优先执行当遇到script标签之后交给JS引擎线程,而它自己则暂停
2.同步代码先执行,遇到定时器就先交给计时器,遇到ajax请求就交给异步请求线程.此时,这两个线程会完成主线程分配的任务,比如计时器开始计时,ajax请求开始请求.然后计时结束或者ajax请求结束时,会把回调函数注册进事件循环列表.
3.当主线程,即JS引擎线程为空时.事件线程会根据事件循环的规则把对应的事件交给JS引擎线程去处理
4.当JS引擎线程的工作结束时,GUI线程开始接手渲染更新过的页面
异步事件注册顺序和回调函数执行顺序不一样
以下面这段代码为例
1 console.log('同步1') 1 2 setTimeout(()=>{ console.log('setTimeout1') 5 },0) //宏任务 3 new Promise((res)=>{
res() console.log('new Promise') //同步 2 4 }).then(()=>{ console.log('Promise.then') //微任务 4 }) 5 console.log('同步2') //同步 3
代码执行顺序:左边的数字(js就是从上往下执行)
函数注册进事件循环列表的顺序: 1 2 3 4 5 或者 1 3 2 4 5 ,因为这两个线程谁先结束谁先注册
事件循环(回调函数执行顺序):右边的数字
注意点:1.js是单线程,但浏览器渲染进程不是不是. 2.异步是目的,是为了不等待而且还能拿到对应的数据.而单线程和事件循环是手段. 3.定时器和异步请求本身是由对应的线程去执行的,但是回调函数会进入事件线程,再由事件线程给JS线程.
4.事件循环是针对回调函数的.