Nodejs既然這么流行就肯定有它的博大精深之處鸳碧,自然不是我這還沒入門的小白可以掌握的盾鳞,我就簡單說一下目前自己的理解程度。
一瞻离、單線程腾仅、非阻塞I/O、事件驅(qū)動(dòng)
這是nodejs的三個(gè)特點(diǎn)套利。
單線程
Nodejs是單線程攒砖,和多線程相比:
優(yōu)點(diǎn):可以避免系統(tǒng)分配多線程以及線程間通信時(shí)的開銷,可以更高效的利用cpu日裙,降低內(nèi)存的耗用。
缺點(diǎn):一旦出現(xiàn)錯(cuò)誤會(huì)導(dǎo)致整個(gè)程序崩潰惰蜜,不擅長大量的計(jì)算昂拂,無法利用多核cpu(貌似現(xiàn)在也有辦法)
非阻塞I/O
非阻塞I/O從名字上便可以理解,為了提高程序的性能抛猖,更好的提高線程的利用率格侯,盡量不讓線程空閑著鼻听。
比如我們想要讀取一個(gè)文件,然后再進(jìn)行一些操作联四,當(dāng)然這些操作的前提條件是不需要這個(gè)文件的數(shù)據(jù)撑碴,這個(gè)時(shí)候我們便可以讓系統(tǒng)的某個(gè)線程去讀取文件,同時(shí)程序的主線程繼續(xù)執(zhí)行下面的操作朝墩,程序并不會(huì)等待文件讀取完畢才繼續(xù)執(zhí)行醉拓,這就像我們常用的ajax
事件驅(qū)動(dòng)
Nodejs提供的絕大多數(shù)API都是基于事件的、異步的風(fēng)格收苏,就是服務(wù)器程序的入口也是從connect事件開始亿卤。
還是繼續(xù)上面的讀取文件的例子,假如我要讀取一個(gè)文件鹿霸,然后要打印出文件的內(nèi)容排吴,那么讀取文件便是一個(gè)事件(readFile),而打印的操作要在事件的回調(diào)函數(shù)中執(zhí)行懦鼠。
在這里又涉及到了事件隊(duì)列和事件循環(huán):
程序主線程順序執(zhí)行钻哩,當(dāng)遇到一個(gè)事件時(shí)會(huì)將其添加進(jìn)事件隊(duì)列,Nodejs底層的libuv庫(不要問我這是什么鬼肛冶,底層的東西看著都嚇人)負(fù)責(zé)將事件隊(duì)列中不同的事件任務(wù)分配給不同的線程去執(zhí)行街氢,執(zhí)行的結(jié)果再重新返回給用戶,這便是事件循環(huán)淑趾。
為什么Nodejs是單線程的阳仔,卻在這里又變成了多線程,其實(shí)Nodejs對(duì)外展示是單線程的扣泊,但內(nèi)部其實(shí)是多線程的近范,Node本身的主線程主要就是起不斷往返調(diào)度的作用
二、例子更好理解
說了這么多延蟹,比個(gè)例子吧评矩。我在網(wǎng)上定了個(gè)外賣,老板在店里安排人給我派送阱飘,定完外賣之后開始打游戲斥杜,正打著激烈的時(shí)候派送員來了,玩的正嗨呢沥匈,沒空開門蔗喂,結(jié)果派送員在門外等了1一分鐘才去開門,至此我定外賣的這個(gè)過程也就結(jié)束了高帖。
在這個(gè)例子中我就相當(dāng)于主線程缰儿,我定了外賣后這個(gè)單子就會(huì)加入到店鋪的訂單列表,就相當(dāng)于一個(gè)事件加入到了事件隊(duì)列散址,老板就是這個(gè)libuv乖阵,他在店里運(yùn)籌帷幄安排人員給我做飯宣赔,然后送外賣,即分配線程執(zhí)行事件任務(wù)瞪浸,當(dāng)送回來時(shí)如果我是空閑的我就會(huì)立刻開門收快遞儒将,但是當(dāng)時(shí)我正忙著(打游戲很重要),所以派送員久等了一會(huì)对蒲,即事件執(zhí)行之后的回調(diào)函數(shù)是否會(huì)立即執(zhí)行要看主線程是否空閑钩蚊。
三、總結(jié)
1.Node 表面上是單線程齐蔽,其實(shí)內(nèi)部仍然是多線程的两疚,主線程起往返調(diào)度
2.單線程、事件驅(qū)動(dòng)含滴、非阻塞I/O,我的理解是事件驅(qū)動(dòng)只是為了實(shí)現(xiàn)非阻塞的方式诱渤,非阻塞才是目的
轉(zhuǎn)載:http://www.reibang.com/p/14bb2b4038d3