单线程的JavaScript如何实现异步操作

说到异步实现,就肯定要说同步,那么首先就来了解一下:

同步&异步


同步:从上往下逐步执行代码,只有上一步完成了,得到结果了,才会进入下一步,也就是上一步对下一步造成了堵塞。

异步:执行之后先不等待结果,直接进行下一步操作

光看文字可能难以理解,举个例子

假如你想做两道菜,一道是爆炒猪肝,从买回原料开始,猪肝先是切成片,很小的片,然后放到一只碗里,放上一些盐,放上生粉,生粉让猪肝鲜嫩。。。。。

最后为了做这道菜你耗费了大量精力,全程都很累

 所以,这个事堵塞了你干别的事,这就是 同步

做菜就这么难吗?其实也不是,还有一道菜:

速食梅菜扣肉,买回来之后放入微波炉,后面就不用管了,期间你完全可以做别的事而不被它影响

这个事不会影响你做别的事,这就是 异步

看看代码


同步代码(代码1)

function someTime() {
    let s = Date.now();
    while(true) {
        if (Date.now() - s > 2000) {
            console.log(2)
            break;
        }
    }
}

console.log(1);
someTime();
console.log(3);

// 打印结果:1 ...(2秒以后)... 2 3

异步代码(代码2)

function someTime() {
    setTimeout(() => {
        console.log(2);
    }, 2000)
}

console.log(1);
someTime();
console.log(3);

// 打印结果:1 3 ...(2秒以后)... 2

结果如上,同步代码在遇到耗时的操作后,后续的代码只会等待,而异步代码遇到耗时的操作后,后续的代码直接掠过,不会等待

JavaScript单线程


众所周知,JavaScript是单线程的语言,所谓单线程,就是指程序执行时,按照从前到后的顺序,一个个的解决问题,必须等前面的执行完,后面的才能开始执行,我们发现,这个解释与 同步 简直如出一辙。

如此看来,JavaScript似乎只能实现同步操作,与异步无缘了~,事实也是如此吗?看如下代码:

function timeOut() {
    setTimeout(() => {
        console.log('time Out');
    }, 0)
}
function someTime() {
    let s = Date.now();
    while(true) {
        if (Date.now() - s > 2000) {
            console.log('some Time')
            break;
        }
    }
}

console.log(1);
timeOut();
someTime();
console.log(3);

按照单线程的思想,上面代码的执行结果应该是 1...time Out......some Time..... 3

然而事实却是 1...some Time...3...time Out

浏览器的多线程


JavaScript作为一种脚本语言,需要一个运行的环境,这个环境就是浏览器(或者node),虽然js是单线程的,但是浏览器却是多线程的。

 JS线程被成为主线程,负责解析js脚本,其他的线程都在辅助他,通过其他线程,js就可以实现异步操作了。

比如代码2中从上往下的代码解析中,遇到了setTimeOut之后,js线程就会把他添加到定时触发器线程中,继续执行下面的代码,完成了异步。

更为重要的是 任务队列 的出现

 任务队列


定时触发器的定时结束后,会将回调函数加入任务队列的队尾,当一个事件(包括定时结束事件)被触发时,也会把回调函数加入任务队列的队尾,等待主线程的处理。

任务添加之后不是直接处理,而是需要等待,等待主线程按顺序进行处理

宏任务&微任务


异步任务还可以细分,分为宏任务(Macrotask 或 tasks)和微任务(Mincrotask 或 jobs) ,结构如下图

首先要知道的是,微任务是要先于宏任务执行的。来段代码 

console.log(1);
setTimeout(() => {
    console.log(2);
}, 0);
Promise.resolve().then(() => {
    console.log(3);
});
console.log(4);

执行结果是 1...4...3...2 ,跟你想的一样吗

可以看到setTimeOut虽然在promise的代码之上,但是却是最后一个执行的,这就是宏任务跟微任务的区别了。至于为什么 ,这个就涉及到浏览器的Event Loop问题了。

浏览器的Event Loop


总结一下浏览器执行js代码的过程:

1.执行全局Script同步代码,形成一个执行栈

2.在执行代码时当遇到如上 异步任务时便会按上文所描述的将宏任务回调加入宏任务队列微任务回调加入任务队列

3.然而,回调函数放入任务队列后也不是立即执行;会等待执行栈中的同步任务全部执行完清空了栈后引擎才能会去任务队列检查是否有任务,如果有那便会将这些任务加入执行栈,然后执行!

4.执行栈清空后,会先去检查微任务队列是否有任务,逐一将其任务加入执行栈中执行,期间如果又产生了微任务那继续将其加入到微任务队列末尾,并在本周期内执行完,直到微任务队列的任务全部 清空,执行栈也清空后,再去检查宏任务队列是否有任务,取到队列队头的任务放入到执行栈中执行,其他可能又会产生微任务,那当本次执行栈中的任务结果清空后又会去检查微任务队列...

5.引擎会循环执行如上步骤,这就是Event Loop!

上代码

console.log('start');
setTimeout(() => {
    console.log('time1');
    Pormise.resolve().then(() => {
        console.log('promise1');
    })
}, 0);
setTimeout(() => {
    console.log('time2');
    Pormise.resolve().then(() => {
        console.log('promise2');
    })
}, 0);
Pormise.resolve().then(() => {
    console.log('promise3');
});
console.log('end');

本文参照大佬,大钟同学的文章编写,共同学习!

单线程的JavaScript是如何实现异步的 - 掘金

  • 6
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JavaScript 是一门单线程的编程语言,意味着它在任意给定的时刻只能执行一个任务。这是因为 JavaScript 在最初设计时是作为浏览器脚本语言而诞生的,用于操作网页的 DOM(文档对象模型)。 在 JavaScript 中,任务按照它们被调用的顺序执行,这种方式称为同步执行。当一个任务执行时,其他任务必须等待它的完成才能继续执行。这种同步执行的特性可以确保数据的一致性,但也可能导致阻塞,特别是在执行耗时较长的任务时。 为了解决阻塞问题,JavaScript 引入了异步执行的概念。通过异步执行,可以让某些任务在后台执行,而不会阻塞其他任务的执行。常见的异步操作包括网络请求、文件读写和定时器等。在 JavaScript 中,通常使用回调函数、Promise、async/await 等方式来处理异步操作。 回调函数是最早被广泛使用的异步处理方式。通过将一个函数作为参数传递给异步操作,在操作完成后调用该函数来处理结果。然而,使用回调函数嵌套多层会导致代码可读性和维护性的降低,这就是所谓的"回调地狱"问题。 为了解决回调地狱问题,Promise 和 async/await 出现了。Promise 是一种用于处理异步操作的对象,它可以链式调用,避免了回调函数嵌套的问题。而 async/await 是基于 Promise 的语法糖,使异步代码看起来更像同步代码,更易于理解和编写。 总结起来,JavaScript单线程的,但通过异步执行可以提高程序的性能和响应速度。同步执行保证了数据的一致性,而异步执行允许在后台处理耗时操作,提高了用户体验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值