首先,我们要知道,JavaScript的本质是一门浏览器脚本语言,在执行的时候是一行一行的执行,只有前面的代码执行完了才会执行后面的代码。JS是单线程语言指的就是这个意思。

同步和异步其实在进行任务执行顺序时候都只有一条流水线,区别在于执行任务的顺序不同。

对于同步任务和异步任务,打个比喻:

有一堆学生在食堂排队打饭,然后进门的时候有些学生领了个异步的牌子,有些学生没有领,然后在窗口前排队打饭的时候,食堂大叔规定,有异步牌子的学生出来重新组成一个小队列,在窗口旁边等待,没有牌子的学生仍然在窗口前的主队里排队打饭。等主队里没有牌子的学生排队打完饭之后,食堂大叔示意小队列里有异步牌子的学生一个个的过来打饭。

抛开这个场景不谈,在js中,同步和异步的概念,有很多博主的解释就是“同步就是任务一个接一个的执行,前面的没有执行完成后面的就一直等待”,“异步就是觉得一个任务要执行会花很长时间,所以先放着,去执行其他的任务,等轮到这个任务的时候再执行。”

很多小伙伴看到这里就懵逼了,怎么着,系统还能识别哪个任务执行时间长哪个短?有这么智能好用的系统?要是系统自动识别执行时长,任务管理不会乱套?所以,我个人觉得这些博主的描述是有很大问题的。

一个任务是否是异步的是看程序员在编写代码的时候是否将这个任务设置为异步,而不是说时间长的任务就一定是异步的,只是通常来讲为了流畅性,编程者会将执行时间长的任务主观的设置为异步。

所以说只要编程者不人为的对任务的执行顺序进行干涉,那么任务就是同步的,会一条一条的执行。但是只要编程者人为的对某些任务的执行顺序进行干涉,就是进行异步操作。

那么这里就涉及到一个问题了,怎么将一个任务设置为异步?

JS中最基础的有两种方式——setTimeout函数和setInterval函数。

如下代码:

 <script type="text/javascript">
console.log('1');
setTimeout(function(){
console.log('2');
},0)
setTimeout(function(){
console.log('3');
},0)
setTimeout(function(){
console.log('4');
},0)
console.log('5');
</script>

上述代码输出顺序为:1  5  2  3  4

其中因为2 3 4的输出语句使用了setTimeout函数将其设定了等待时间(虽然设置的等待时间为0),所以它们3个被抽出来放在了另外的队列里,只有没被特别设置的任务按顺序执行完之后才会开始执行这个特殊队列的任务。所以会先把正常的1 5打印,然后才会开始打印2 3 4

这里我们会发现,1和5是按一般顺序打印,因为JS是逐行执行的。但是同为异步任务的2 3 4的打印任务也是按照2 3 4的顺序打印,是受加入队列的顺序影响。那么是不是说异步任务组成的队列里执行顺序和加入异步队列的顺序有关呢?并不是。只有异步任务的等待时间相同的时候,异步任务的执行顺序才会收到计入队列的顺序影响。否则的话主要还是与每个任务设置的等待时间有关系。

例如下面这种情况:

 <script type="text/javascript">
console.log('1');
setTimeout(function(){
console.log('2')
},300)
setTimeout(function(){
console.log('2')
},200)
setTimeout(function(){
console.log('2')
},100)
console.log('5');
</script>

与等待时间有关

上述代码的输出顺序为1  5  4  3  2

因为异步任务设置的等待时间不同,在同步任务执行完后,等待时间较短的任务优先执行,等待时间长的后执行。

把这些都搞清楚之后我们又难免会思考,这样的异步操作都是对于只需要执行一次的任务进行排序,假如是要对多个需要按一定顺序循环执行的异步任务呢?这个时候就需要用到promise队列的链式调用,并且使用一个全局变量,在每完成一次循环后重置全局变量的值,并且在循环执行前检查该全局变量的值。

详细思路和是西安方法可以参考以下这篇博文。

关于JS下大批量异步任务按顺序执行解决方案

JS中同步和异步的更多相关文章

  1. js中同步与异步处理方法

    在使用异步请求时,有时需要将异步请求的结果返回给另一个js函数,此种情况下会出现未等异步请求返回请求结果,该发送请求所在js函数已经执行完后续操作,即已经执行return ,这样会导致return的结 ...

  2. js中同步与异步请求方式

    异步请求方式: $.ajax({ url : 'your url', data:{name:value}, cache : false, async : true, type : "POST ...

  3. JS中同步与异步

    不讲过多定义,举两个例子说明下 例一: console.log(100); setTimeout(function(){ console.log(200); },1000); console.log( ...

  4. 在JavaScript中同步与异步

    在JavaScript中,一个线程执行的时候不依靠其他线程处理完毕我们称为异步,相反一个线程必须等待直到另一个线程处理完毕我们则称为同步.打个比方: (1)同步就是你在煮方便面的时候必须等水开了,你才 ...

  5. JS中同步与异步的理解

    你应该知道,javascript语言是一门“单线程”的语言,不像java语言,类继承Thread再来个thread.start就可以开辟一个线程,所以,javascript就像一条流水线,仅仅是一条流 ...

  6. HTTP请求中同步与异步有什么不同

    普通的B/S模式就是同步,而AJAX技术就是异步,当然XMLHttpReques有同步的选项. 同步:提交请求->等待服务器处理->处理完毕返回.这个期间客户端浏览器不能干任何事. 异步: ...

  7. JS的同步和异步加载

    引言 JS的“加载”不能理解为下载,它是分为两个部分:下载,执行.默认的JS加载是同步的,因为浏览器需要一个稳定的DOM结构,而执行JS时可能会对DOM造成改变,所以在执行JS时一定会阻塞HTML的渲 ...

  8. js的同步与异步

    JavaScript语言的一大特点就是单线程,也就是说,同一个时间只能做一件事.那么,为什么JavaScript不能有多个线程呢?这样能提高效率啊. JavaScript的单线程,与它的用途有关.作为 ...

  9. IO中同步、异步与阻塞、非阻塞的区别

    一.同步与异步同步/异步, 它们是消息的通知机制 1. 概念解释A. 同步所谓同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不返回. 按照这个定义,其实绝大多数函数都是同步调用(例如si ...

随机推荐

  1. BigDecimal的setScale常用方法(ROUND_UP、ROUND_DOWN、ROUND_HALF_UP、ROUND_HALF_DOWN)

    BigDecimal的setScale四大常用方法总结 // 设置小数点后第三位数字一大一小观察效果BigDecimal num = new BigDecimal("3.3235667&qu ...

  2. Python 图像处理 OpenCV (7):图像平滑(滤波)处理

    前文传送门: 「Python 图像处理 OpenCV (1):入门」 「Python 图像处理 OpenCV (2):像素处理与 Numpy 操作以及 Matplotlib 显示图像」 「Python ...

  3. AS中将module转成library的步骤

    转换步骤是在Android Studio 2.3版本下进行的,其他版本未测试 将要变成library的module的gradle文件的第一行 修改前:apply plugin: 'com.Androi ...

  4. 使用vw进行移动端适配(nuxt项目)

    基于nuxt 2.0.0 一.安装postcss-px-to-viewport npm安装 npm install postcss-px-to-viewport --save-dev 或 yarn安装 ...

  5. windows server2012 安装SQL SERVER 2016环境监测出错

    Windows Server 2012 R2 安装SQL Server 2016 顺序为:KB2919442 ——> KB2919355 ——> SQL Server 2016 并且还要安 ...

  6. 文本溢出后,隐藏显示"..."和margin边距重叠

    一.隐藏加省略 单行文本: overflow: hidden; white-space: nowrap; text-overflow: ellipsis; 多行文本: overflow: hidden ...

  7. 【JMeter_01】JMeter介绍与环境搭建

    JMeter介绍 Apache JMeter™应用开源软件,100%纯Java应用程序,设计之初是用于负载功能测试和性能测试.但因它在实现对各种接口的调用方面比较成熟,因此,常被用做接口功能测试. J ...

  8. c++ cc24a_demo //转换函数,用来做转换操作符,int()括号里面必须是空的,必须定义为const,代码示范

    c++ cc24a_demo //转换函数,用来做转换操作符,int()括号里面必须是空的,必须定义为const,代码示范 #include <iostream> #include < ...

  9. ASP.NET Core 对Controller进行单元测试

    单元测试对我们的代码质量非常重要.很多同学都会对业务逻辑或者工具方法写测试用例,但是往往忽略了对Controller层写单元测试.我所在的公司没见过一个对Controller写过测试的.今天来演示下如 ...

  10. Centos中使用virtualenvwrapper

    Centos中使用virtualenvwrapper python特有的一种软件环境,创建多个python环境,各个环境之间完全隔离,互不影响.它可以用来解决Python项目开发和运行过程中的依赖项和 ...