js的单线程,异步及回调函数

最近本人对于js的运行机制,特别是异步,还有回调函数感觉很乱,于是参考了很多有用的博客(博客原文地址会在文末给出),整理如下:


js单线程

我们都知道,Javascript语言的执行环境是"单线程"(single thread)。也就是说,浏览器只分配给js一个主线程用来执行任务即函数,但是每次只能执行一个任务,只有等到当前任务执行完成后,才执行后面的任务,这些任务形成一个任务队列排队等候执行,这一点和我们日常的排队很像,譬如排队买奶茶,只有等到前面一个人买完奶茶付完钱,排在他后面的人才可以买奶茶。但是,当前面一个任务很耗时时,后面的任务就不得不等着,这时候整个程序的执行效率就会下降,就像我们平时遇到的浏览器无响应即页面假死往往是因为某段js代码长时间运行如死循环,导致页面卡死,后面的任务无法执行。

讲到js的单线程,就不得不来了解一下浏览器

浏览器多线程

浏览器主要线程

如图,浏览器是一个多线程的执行环境,在浏览器的内核中分配了多个线程,其中浏览器常驻三大线程: js引擎线程,GUI渲染线程,浏览器事件触发线程。最主要的线程之一即是js引擎的线程。由于这三个线程同时要访问DOM树,所以为了线程安全,浏览器内部需要做互斥即当JS引擎在执行代码的时候,界面渲染和事件响应两个线程是被暂停的。而所以当JS出现死循环,浏览器无法响应点击,也无法更新界面。

前面说到,前端会有一些任务十分耗时,而由于js是单线程使用会降低执行效率,这些耗时的任务如网络请求,定时器和事件监听。所以,浏览器为这些耗时任务开辟了另外的线程,主要包括http请求线程,浏览器定时触发器,浏览器事件触发线程,这些任务是异步的(见图)。(详细过程见此博文,博主讲得很好~ 个人建议必须看一看哦~)

任务队列

说回刚刚的js单线程,为了不让前面的耗时任务导致的问题出现,js的设计者把js的任务分为同步任务和异步任务。同步任务指的是,在主线程上排队执行的任务,只有前一个任务执行完毕,才能执行后一个任务;异步任务指的是,不进入主线程、而进入"任务队列"(task queue)的任务,只有"任务队列"通知主线程,某个异步任务可以执行了,该任务才会进入主线程执行。如我们刚刚所讲到的浏览器为网络请求开辟的http请求线程就是异步任务。

具体来说,异步执行的运行机制如下。(同步执行也是如此,因为它可以被视为没有异步任务的异步执行。)

(1)所有同步任务都在主线程上执行,形成一个[执行栈]。
(2)主线程之外,还存在一个"任务队列"(task queue)。只要异步任务有了运行结果,就在"任务队列"之中放置一个事件。
(3)一旦"执行栈"中的所有同步任务执行完毕,系统就会读取"任务队列",看看里面有哪些事件。那些对应的异步任务,于是结束等待状态,进入执行栈,开始执行。
(4)主线程不断重复上面的第三步。

(参考与阮一峰老师的博文JavaScript 运行机制详解:再谈Event Loop

回调函数

有了以上了解我们可以知道,主线程内的同步任务执行完毕后,就会执行排在任务队列第一位的异步任务,这个过程不断重复。

当主线程开始执行异步任务,实际就是执行对应的回调函数。

我们来看一下例子:

setTimeout(function(){
    console.log('Hello');
},10);

执行这段代码,浏览器异步执行计时操作(注意这里的浏览器模型定时计数器并不是由JavaScript引擎计数的),当10ms到了之后,就会触发定时事件,这时就会把其中的回调函数放到任务队列中,所以当主线程空闲时在任务队列中“读取”并且执行的就是回调函数。

异步任务必须指定回调函数。

Event Loop

主线程从"任务队列"中读取事件,这个过程是循环不断的,所以整个的这种运行机制又称为Event Loop(事件循环)。

图片来自Philip Roberts的演讲《Help, I'm stuck in an event-loop》

如图,WebAPIs就是js线程外部的api如我们刚刚所说浏览器为异步任务所开辟的线程。而任务队列就是callbackqueue,我们知道任务队列中其实就是各种异步任务的回调函数,从callbackqueue的直译“回调队列”也可看出。而heap堆和stack栈组成了js的主线程,当stack中的函数执行完成后,就会在callbackqueue中寻找下一个任务并把它推入栈,这个寻找的过程就叫event loop(事件循环)。


看了上面你是不是对js的运行机制有了了解呢~
我们把学会的知识来用一用:

js中的异步之定时器

说起js的异步,很多人第一反应是Ajax,但其实js中最基础的异步就是setTimeout/setInterval。(小伙伴可不要把定时器忘了哦:))

我们以setTimeout为例,setTimeout接受两个参数,第一个是回调函数,第二个是推迟执行的毫秒数。

我们看看例子:

setTimeout(function(){
  console.log(0);
},0)
 
console.log(1);
 
// 1
 
// 0

是不是以为打印的顺序是0,1?但大家注意哦,这时候浏览器打印的顺序是1,0。大家可能疑问了,setTimeout中设置的推迟执行的毫秒数是0呀,不就是立即执行的意思吗。大家还记得刚刚我们说了当有耗时任务时,会把它放在任务队列中等待主线程空闲然后再执行,实际在执行程序的时候,浏览器会默认setTimeout以及ajax请求这一类的方法都是耗时程序(尽管可能不耗时),也就是上面说过的浏览器会为其异步开辟线程。所以此时的setTimeout尽管它推迟时间为0,但是js不会立即执行,而是把它加入任务队列,当执行完执行栈的同步任务也就是打印1后,再执行setTimeout的回调函数,打印0。

总之,setTimeout(fn,0)的含义是,指定某个任务在主线程最早可得的空闲时间执行,也就是说,尽可能早得执行。它在"任务队列"的尾部添加一个事件,因此要等到同步任务和"任务队列"现有的事件都处理完,才会得到执行。

所以注意的是,setTimeout()只是将事件插入了任务队列,必须等到当前代码(执行栈)执行完,主线程才会去执行它指定的回调函数。但如果当前任务十分耗时,需要等很久,所以并没有办法保证,回调函数一定会在setTimeout()指定的时间执行,比如说你指定10ms后执行,但是当前的任务执行了20ms,所以setTimeout的回调函数并不能在10ms后立即执行,可能要20ms后,如果setTimeout在任务队列中不是排第一位,可能还不止20ms。

js异步编程的方法

这个我还不是很懂~大家可以参考阮一峰老师的Javascript异步编程的4种方法

希望一包的文章可以帮到你们~


参考文章:

http://blog.csdn.net/qq_22855325/article/details/72958345(赞~)
https://www.cnblogs.com/woodyblog/p/6061671.html(重点参考这篇文章博主写得很好~)
http://blog.csdn.net/kfanning/article/details/5768776(赞~)
http://www.ruanyifeng.com/blog/2014/10/event-loop.html(阮一峰老师嘛~)
http://www.cnblogs.com/smght/p/4368399.html#3575993(赞~)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容

  • ———-正文开始———- 最近发现有不少介绍JS单线程运行机制的文章,但是发现很多都仅仅是介绍某一部分的知识,而且...
    流动码文阅读 2,192评论 7 32
  • 弄懂js异步 讲异步之前,我们必须掌握一个基础知识-event-loop。 我们知道JavaScript的一大特点...
    DCbryant阅读 2,708评论 0 5
  • 1.背景介绍 什么是同步,什么是异步? 同步指的是一次只能完成一件任务。如果有多个任务,就必须排队,前面一个任务完...
    风流司郎中阅读 720评论 1 6
  • 我离开家已有十六年了,十六年来只回过四次家,也都是行色匆匆,没有在老屋睡上一次觉,更没有陪着娘坐上一天。 我少年睡...
    永无休止阅读 1,159评论 13 18
  • 凌晨两点,听灵魂哭泣 染了血色的雨水,斑驳了俊颜 墨色丝线,断了一根一根 被扼住的喉咙,发出沙哑嘶鸣 亲手剜下心脏...
    山半阅读 428评论 37 24