詳解node單線程實(shí)現(xiàn)高并發(fā)原理與node異步I/O
一、node單線程實(shí)現(xiàn)高并發(fā)原理
眾所周知nodejs是單線程且支持高并發(fā)的腳本語言??蔀槭裁磫尉€程的nodejs可以支持高并發(fā)呢?很多人都不明白其原理,下面我來談?wù)勎业睦斫猓?/p>
1. node的優(yōu)點(diǎn):I/O密集型處理是node的強(qiáng)項(xiàng),因?yàn)閚ode的I/O請求都是異步的(如:sql查詢請求、文件流操作操作請求、http請求...)
a. 什么是異步?
異步:發(fā)出操作指令,然后就可以去做別的事情了,所有操作完成后再執(zhí)行回調(diào)
異步的實(shí)現(xiàn)原理:
// 第一步:定義變量 let a = 1; // 第二步:發(fā)出指令,然后把回調(diào)函數(shù)加入異步隊(duì)列(回調(diào)函數(shù)并沒有執(zhí)行) setTimeout(() => { console.log(a); }, 0) // 第三步:賦值,回調(diào)函數(shù)沒有執(zhí)行 a = 2; // 第四步:發(fā)出指令,然后把回調(diào)函數(shù)加入異步隊(duì)列(回調(diào)函數(shù)并沒有執(zhí)行) setTimeout(() => { console.log(a); }, 0) // 第五步:賦值,回調(diào)函數(shù)沒有執(zhí)行 a = 3; // 當(dāng)所有代碼執(zhí)行完畢,cpu空閑下來了,就會(huì)開始執(zhí)行異步隊(duì)列里面的回調(diào)函數(shù) // 所以最后控制臺(tái)輸出:3 3
b. 什么是異步I/O?
異步I/O顧名思義就是異步的發(fā)出I/O請求
c. 雖然nodejs可以異步的發(fā)出I/O請求,但nodejs不支持多線程,為啥就可以支持高并發(fā)呢?
因?yàn)閚odejs的I/O操作,底層是開啟了多線程的
當(dāng)同時(shí)有多個(gè)IO請求時(shí),主線程會(huì)創(chuàng)建多個(gè)eio線程,以提高IO請求的處理速度
額外知識(shí)點(diǎn):
d. 雖然nodejs的I/O操作開啟了多線程,但是所有線程都是基于主線程開啟的只能跑在一個(gè)進(jìn)程當(dāng)中還是不能充分利用cpu資源
pm2進(jìn)程管理器可以解決這個(gè)問題
pm2 是一個(gè)帶有負(fù)載均衡功能的Node應(yīng)用的進(jìn)程管理器.
e. cpu核數(shù)與線程之間的關(guān)系
在過去單CPU時(shí)代,單任務(wù)在一個(gè)時(shí)間點(diǎn)只能執(zhí)行單一程序。之后發(fā)展到多任務(wù)階段,計(jì)算機(jī)能在同一時(shí)間點(diǎn)并行執(zhí)行多任務(wù)或多進(jìn)程。雖然并不是真正意義上的“同一時(shí)間點(diǎn)”,而是多個(gè)任務(wù)或進(jìn)程共享一個(gè)CPU,并交由操作系統(tǒng)來完成多任務(wù)間對CPU的運(yùn)行切換,以使得每個(gè)任務(wù)都有機(jī)會(huì)獲得一定的時(shí)間片運(yùn)行。而現(xiàn)在多核CPU的情況下,同一時(shí)間點(diǎn)可以執(zhí)行多個(gè)任務(wù),具體到這個(gè)任務(wù)在CPU哪個(gè)核上運(yùn)行,這個(gè)就跟操作系統(tǒng)和CPU本身的設(shè)計(jì)相關(guān)了
2. node的缺點(diǎn):不擅長cpu密集型的操作
a. 什么是cpu密集型操作(復(fù)雜的運(yùn)算、圖片的操作)
// 這就是一個(gè)cpu密集型的操作 for (let i = 0; i < 1000000; i++) { console.log(i); }
b. nodejs為什么不擅長cpu密集型操作
因?yàn)閚odejs是單線程的
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
nodejs使用Sequelize框架操作數(shù)據(jù)庫的實(shí)現(xiàn)
這篇文章主要介紹了nodejs使用Sequelize框架操作數(shù)據(jù)庫的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10express框架+bootstrap美化ejs模板實(shí)例分析
這篇文章主要介紹了express框架+bootstrap美化ejs模板,結(jié)合實(shí)例形式分析了express框架引入bootstrap結(jié)合ejs模版引擎相關(guān)操作技巧與注意事項(xiàng),需要的朋友可以參考下2023-05-05nodejs實(shí)現(xiàn)郵件發(fā)送服務(wù)實(shí)例分享
本文給大家講解的是簡單的使用nodejs搭建郵件發(fā)送服務(wù)的一個(gè)實(shí)例,非常的好用,有需要的小伙伴可以參考下2017-03-03Node.js環(huán)境下JavaScript實(shí)現(xiàn)單鏈表與雙鏈表結(jié)構(gòu)
Node環(huán)境下通過npm可以獲取list的幾個(gè)相關(guān)庫,但是我們這里注重于自己動(dòng)手實(shí)現(xiàn),接下來就一起來看一下Node.js環(huán)境下JavaScript實(shí)現(xiàn)單鏈表與雙鏈表結(jié)構(gòu)2016-06-06nodejs中函數(shù)的調(diào)用實(shí)例詳解
本文通過實(shí)例代碼給大家介紹了nodejs函數(shù)的調(diào)用,代碼簡單易懂,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-10-10node.js中使用q.js實(shí)現(xiàn)api的promise化
這篇文章主要介紹了node.js中使用q.js實(shí)現(xiàn)api的promise化,promise一個(gè)標(biāo)準(zhǔn),它描述了異步調(diào)用的返回結(jié)果,包括正確返回結(jié)果和錯(cuò)誤處理,需要的朋友可以參考下2014-09-09