js面试题之异步问题的深入理解


Posted in Javascript onSeptember 20, 2020

js中的宏任务与微任务

在面试过程中,基本面试官都会问你一些promise的问题,promise是es6的新内容,主要是用来优化异步的问题。笔试中经常会让你写一些promise和setTimeout的执行结果,这你就必须知道宏任务和微任务的概念了!

为什么要使用promise

如果你经历过以前的jquery开发项目,你会遇到以下问题:回调地狱

$.ajax({
	...
	success: function() {
		...
		$.ajax({
			...
			success: function() {
				
			}
		})
		...
	}
})

原因分析:

ajax请求嵌套,原因是我第二个请求依赖的参数在第一个请求的结果中,所以就得这么一直嵌套下去,ajax是异步的,不能再外面拿到里面的结果。这种代码导致的问题就是调试困难,耦合性非常高,后期改动一个地方就头疼!维护非常困难,代码可读性差。

于是乎就引入了promise对ajax进行了优化,axios就是基于promise的一个请求封装库,他们底层都是基于js原生的XMLHTTPREQUEST.

promise().then().catch()链式调用,多个请求可以promise().then().then()。

何为宏任务,何为微任务?

思考这个问题时你必须知道javascript是一种单线程的脚本语言,也就是它的代码正常只能从上往下依次执行,一次只能做一件事,异步是通过回调函数来实现的。为何不把js设计成多线程的语言呢?语言的用途决定了它的特性,js最初是用来做表单验证以及正则判断的,和操作DOM元素的。如果js有多个线程,一个执行DOM元素修改,另一个执行删除,那浏览器直接懵逼了,我到底该干啥???所以语言的用途决定了他的特性,但是浏览器是多线程的,除了主线程还有其他线程。

当js主程序执行时,先运行主程序上的同步代码,遇到setTimeout或setInterval就把它放入宏队列中,遇到promise的回调就把它放到微队列中,程序执行先执行主程序代码,再执行nextTick代码,然后微任务,最后宏任务,任务队列中的依次排队执行,async和await是配套使用的,await后面接一个promise对象,来看看下面这段代码:

setTimeout(function(){console.log(1)},0); // 进入宏任务队列,最后执行宏任务
 new Promise(function(resolve,reject){
   console.log(2); //这句代码在promise构造器,同步执行
   resolve(); // 执行了resolve再把任务放入微队列
 }).then(function(){console.log(3)
 }).then(function(){console.log(4)});
 process.nextTick(function(){console.log(5)});
 console.log(6); // 主程序代码
 // 输出2,6,5,3,4,1
 
// 下面这个进阶代码
setTimeout(function(){console.log(1)},0); // 进入宏任务排序为1
new Promise(function(resolve,reject){
   console.log(2);
   // promise中执行完resolve()才会执行then(),而这里的resolve在宏任务里,执行完主程序代码后,还得先执行先进入宏队列中的程序
   setTimeout(function(){resolve()},0) // 进入宏任务排序为2
 }).then(function(){console.log(3)
 }).then(function(){console.log(4)});
 process.nextTick(function(){console.log(5)});
 console.log(6);
 // 输出的是 2 6 5 1 3 4

再看async and await中的执行顺序

代码如下(示例):

async function async1() {
  console.log(1); 
  await async2();
  console.log(2); //这里要等await执行成功才会执行,进入微任务,排序1
}
async function async2() {
  console.log(3);
}
console.log(4); //主程序代码
setTimeout(function() {
  console.log(5);
}, 0) //进入宏任务,最后执行
async1();
new Promise(function(resolve) {
  console.log(6); // 这句同步执行
  resolve(); 
}).then(function() {
  console.log(7); //进入微任务,排序2
});
console.log(8); // 主程序代码
// 输出的是 4,1,3,6,8,2,7,5

总结

js是单线程语言,它的用途决定了他的特性,异步操作通过事件循环机制,先执行同步代码,然后微任务,最后宏任务,两个任务队列里的任务排队依次执行。await后面的代码必须等待promise返回结果再执行下面代码,await和async是generator函数的语法糖。

到此这篇关于js面试题之异步问题的文章就介绍到这了,更多相关js面试题之异步内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
baidu博客的编辑友情链接的新的层窗口!经典~支持【FF】
Feb 09 Javascript
IE和Firefox下javascript的兼容写法小结
Dec 10 Javascript
浅析return false的正确使用
Nov 04 Javascript
jQuery实现分章节锚点“回到顶部”动画特效代码
Oct 23 Javascript
JavaScript直播评论发弹幕切图功能点集合效果代码
Jun 26 Javascript
利用jQuery插件imgAreaSelect实现图片上传裁剪(放大缩小)
Dec 02 Javascript
移动端web滚动分页的实现方法
May 05 Javascript
深入理解vue-loader如何使用
Jun 06 Javascript
jQuery列表检索功能实现代码
Jul 17 jQuery
使用JavaScript进行表单校验功能
Aug 01 Javascript
vue实现微信二次分享以及自定义分享的示例
Mar 20 Javascript
微信小程序结合Storage实现搜索历史效果
May 18 Javascript
js实现简单的点名器随机色实例代码
Sep 20 #Javascript
前端vue+elementUI如何实现记住密码功能
Sep 20 #Javascript
Vue+element+cookie记住密码功能的简单实现方法
Sep 20 #Javascript
解决vue项目运行提示Warnings while compiling.警告的问题
Sep 18 #Javascript
vue-cli3 热更新配置操作
Sep 18 #Javascript
vue-cli 关闭热更新操作
Sep 18 #Javascript
Node.JS如何实现JWT原理
Sep 18 #Javascript
You might like
攻克CakePHP系列二 表单数据显示
2008/10/22 PHP
php实现按指定大小等比缩放生成上传图片缩略图的方法
2014/12/15 PHP
WordPress中编写自定义存储字段的相关PHP函数解析
2015/12/25 PHP
PHP信号量基本用法实例详解
2016/02/12 PHP
Yii2中添加全局函数的方法分析
2017/05/04 PHP
thinkPHP实现基于ajax的评论回复功能
2018/06/22 PHP
用javascript父窗口控制只弹出一个子窗口
2007/04/10 Javascript
JavaScript 密码强度判断代码
2009/09/05 Javascript
document.onreadystatechange事件的用法分析
2009/10/17 Javascript
鼠标滚轴控制文本框值的JS代码
2013/11/19 Javascript
PHP结合jQuery实现红蓝投票功能特效
2015/07/22 Javascript
简单的vue-resourse获取json并应用到模板示例
2017/02/10 Javascript
Bootstrap显示与隐藏简单实现代码
2017/03/06 Javascript
详解angular笔记路由之angular-router
2017/09/12 Javascript
React Native模块之Permissions权限申请的实例相机
2017/09/28 Javascript
js判断数组是否包含某个字符串变量的实例
2017/11/24 Javascript
微信小程序实现搜索功能并跳转搜索结果页面
2019/05/18 Javascript
vue获取验证码倒计时组件
2019/08/26 Javascript
vue实现抖音时间转盘
2019/09/08 Javascript
JavaScript事件循环及宏任务微任务原理解析
2020/09/02 Javascript
[01:15:44]首部DOTA2纪录片今日23时全网上映
2014/03/19 DOTA
python判断windows隐藏文件的方法
2014/03/21 Python
Python使用函数默认值实现函数静态变量的方法
2014/08/18 Python
Python3实现将文件归档到zip文件及从zip文件中读取数据的方法
2015/05/22 Python
selenium跳过webdriver检测并模拟登录淘宝
2019/06/12 Python
利用python list完成最简单的DB连接池方法
2019/08/09 Python
python使用sklearn实现决策树的方法示例
2019/09/12 Python
深入浅析python的第三方库pandas
2020/02/13 Python
Django如何使用jwt获取用户信息
2020/04/21 Python
python多进程 主进程和子进程间共享和不共享全局变量实例
2020/04/25 Python
CSS3 Media Queries(响应式布局可以让你定制不同的分辨率和设备)
2013/06/06 HTML / CSS
毕业生自荐书
2014/02/03 职场文书
彩色的翅膀教学反思
2014/04/25 职场文书
学雷锋活动倡议书
2014/08/30 职场文书
党的群众路线教育实践活动学习计划
2014/11/03 职场文书
房屋维修申请报告
2015/05/18 职场文书