Node.js中同步和異步編程的區(qū)別及使用方法
一、進(jìn)程和線程
代碼編寫完畢在編譯的過程中計(jì)算機(jī)的內(nèi)存中會(huì)開辟一個(gè)空間來存儲(chǔ)代碼,這個(gè)空間就相當(dāng)于是進(jìn)程,可以將進(jìn)程類比于工廠的廠房,但代碼相當(dāng)于原材料,但僅有廠房和原材料無法生產(chǎn),還需要工人進(jìn)行加工,工人則類比于線程
- 線程(廠房):程序運(yùn)行的環(huán)境
- 線程(工人):線程
二、同步和異步
程序的運(yùn)行分為同步和異步兩種;
同步
同步指的是事情按照順序執(zhí)行;例如早上起床,我們先刷牙,再洗臉,最后吃早餐,講究的是事情先后的執(zhí)行順序
通常情況下我們的程序時(shí)按照同步執(zhí)行的,即按照順序一條一條執(zhí)行,一條執(zhí)行完畢執(zhí)行下一條,如下所示:
/*
同步
- 通常情況下代碼都是自上而下的
*/
console.log("哈哈")
console.log("嘻嘻")
console.log("嘿嘿")通過快捷鍵F5通過NodeJs在控制臺(tái)中打印輸出內(nèi)容為以上順序執(zhí)行的結(jié)果
哈哈
嘻嘻
嘿嘿
下面的代碼示例就是非常典型的同步結(jié)構(gòu)的代碼:
function sum(a, b){
return a + b
}
console.log("第一行打印")
let result = sum(123, 234)
console.log(result)
console.log("第二行打印")同步的特點(diǎn)是邏輯清晰、結(jié)構(gòu)簡單、容易理解;
同步:
- 通常情況下代碼都是自上而下一行一行執(zhí)行的
- 前邊的代碼不執(zhí)行后邊的代碼也不會(huì)執(zhí)行
- 同步的代碼執(zhí)行會(huì)出現(xiàn)阻塞的情況
- 一行代碼執(zhí)行慢會(huì)影響到整個(gè)代碼的執(zhí)行
解決同步的問題:
java python
- 通過多線程來解決
node.js
- 通過異步的方式來解決
阻塞
同步最大的問題就是阻塞;
所謂的阻塞就是按順序執(zhí)行的代碼,前面不執(zhí)行完畢后面的代碼不會(huì)執(zhí)行;
function sum(a, b){
let begin = Date.now()
while(Date.now() - begin < 10000){
}
return a + b
}
console.log("第一行打印")
let result = sum(123, 234)
console.log(result)
console.log("第二行打印")上述sum()執(zhí)行的時(shí)候會(huì)停頓10秒,10秒后才會(huì)返回結(jié)果,由于是同步執(zhí)行代碼,所以sum()會(huì)阻礙后面所有代碼的執(zhí)行,導(dǎo)致整個(gè)程序的執(zhí)行速度變慢
異步
- 一段代碼的執(zhí)行不會(huì)影響到其他程序
- 異步的問題:
- 異步的代碼無法通過return來設(shè)置返回值
- 特點(diǎn):
1. 不會(huì)阻塞其他代碼的執(zhí)行
2. 需要通過回調(diào)函數(shù)來返回結(jié)果
- 基于回調(diào)函數(shù)的異步帶來的問題
1. 代碼的可讀性差
2. 可調(diào)試性差
- 解決問題:
- 需要一個(gè)東西,可以代替回調(diào)函數(shù)來給我們返回結(jié)果
- Promise是一個(gè)可以用來存儲(chǔ)數(shù)據(jù)的對象
Promise存儲(chǔ)數(shù)據(jù)的方式比較特殊;
這種特殊的方式使得Promise可以用來存儲(chǔ)異步調(diào)用的數(shù)據(jù)
在程序中,有些代碼的執(zhí)行速度很快,比如說:打印一個(gè)內(nèi)容、接受一個(gè)請求、發(fā)送一個(gè)響應(yīng)這些都是簡單且能快速反應(yīng)的操作,但例如:讀寫硬盤中的文件(I/O)操作這些就會(huì)非常的慢,如果在這些非常耗時(shí)的操作影響到能快速反應(yīng)的操作,這是我們不希望看到的
對于其他語言,例如java,處理這類問題的方式簡單粗暴,就是直接開啟多線程,每一個(gè)線程處理一個(gè)事務(wù),避免相互干擾;
但是對于Node.js來說,它本身就是單線程的,那么如何處理這種情況,答案就是采用異步;當(dāng)我們?nèi)プx取一個(gè)比較大的文件時(shí),node先將指令發(fā)給計(jì)算機(jī),然后再由計(jì)算機(jī)去讀取文件,此時(shí)node的線程不是等待計(jì)算機(jī)數(shù)據(jù)返回,而是繼續(xù)向下執(zhí)行其他的操作,那么何時(shí)去獲取計(jì)算機(jī)讀取到的數(shù)據(jù)呢?等數(shù)據(jù)返回了我們再去讀取,這樣既不影響其他操作也可以正常的讀取到計(jì)算機(jī)上返回的數(shù)據(jù);
function sum(a, b){
setTimeout(()=>{
return a + b
}, 10000)
}
console.log("第一行打印")
let result = sum(123, 234)
console.log(result)
console.log("第二行打印")上述代碼中,我們將計(jì)算操作放到了setTimeout中,同樣是等待10s,但是setTimeout不會(huì)阻塞其他代碼的執(zhí)行,而是在10s之后將函數(shù)放到了任務(wù)隊(duì)列,這樣一來就可以很好的解決阻塞問題。
但由于函數(shù)的返回值return到了setTimeout的回調(diào)函數(shù)中,此時(shí)我們再調(diào)用sum() 就無法獲取到函數(shù)的計(jì)算結(jié)果了,因此后面調(diào)用sum()傳入實(shí)參的時(shí)候計(jì)算結(jié)果為undefined;
那么如何獲取異步代碼的執(zhí)行結(jié)果呢,只有通過回調(diào)函數(shù)了,異步代碼通常都需要一個(gè)回調(diào)函數(shù)作為參數(shù),當(dāng)異步代碼執(zhí)行完畢取得結(jié)果時(shí)候便可以將結(jié)果作為回調(diào)函數(shù)的參數(shù)進(jìn)行傳遞,這樣我們便可以在回調(diào)函數(shù)中來讀取結(jié)果,并完成后續(xù)操作
// 回調(diào)函數(shù):將函數(shù)作為參數(shù)傳遞
function sum(a, b, cb){
setTimeout(()=>{
cb(a + b)
}, 10000)
}
console.log("第一行打印")
sum(123, 234, result => {
console.log(result)
})
console.log("第二行打印")問題
異步通過回調(diào)函數(shù)解決運(yùn)算結(jié)果的傳遞問題,但最大的問題也來自于回調(diào)函數(shù),由于是異步執(zhí)行,回調(diào)函數(shù)無法直接通過返回值來返回執(zhí)行結(jié)果,想要取得結(jié)果必須通過回調(diào)函數(shù),
這樣就帶來一個(gè)問題:如果我們有兩個(gè)異步操作需要先后執(zhí)行,一個(gè)異步操作依賴于上一個(gè)異步操作的結(jié)果,那么我們只能采取嵌套措施了
// 回調(diào)函數(shù):將函數(shù)作為參數(shù)傳遞
function sum(a, b, cb){
setTimeout(()=>{
cb(a + b)
}, 10000)
}
console.log("11111")
sum(123, 234, result => {
// 計(jì)算777與前兩個(gè)數(shù)的加和
sum(result, 7, result => {
// 拿結(jié)果再進(jìn)行求和操作
sum(result, 8, result => {
sum(result, 9, result => {
})
})
})
})上述的示例中,調(diào)用了4次sum,每一次都調(diào)用了之前的計(jì)算結(jié)果,后一個(gè)是依賴前運(yùn)算結(jié)果的;
這樣子就是“回調(diào)地獄",又名死亡金字塔,這還只是4次,現(xiàn)實(shí)的代碼可能比這更加復(fù)雜;
總之,異步提高了代碼運(yùn)行的效率,同樣也增加了代碼的復(fù)雜程度;
到此這篇關(guān)于Node.js中同步和異步編程的區(qū)別及使用方法的文章就介紹到這了,更多相關(guān)Node.js中同步和異步內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Nodejs搭建多進(jìn)程Web服務(wù)器實(shí)現(xiàn)過程
這篇文章主要為大家介紹了Nodejs搭建多進(jìn)程Web服務(wù)器實(shí)現(xiàn)過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10
npm安裝sharp出現(xiàn)的問題詳解(安裝失敗的問題及解決)
這篇文章主要給大家介紹了關(guān)于npm安裝sharp出現(xiàn)的問題(安裝失敗的問題及解決)的相關(guān)資料,sharp包是基于node.js的高性能圖片處理器,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-11-11
詳解如何在nodejs項(xiàng)目中使用Vue的響應(yīng)式API
vue3的響應(yīng)式API大家應(yīng)該都特別熟悉,平時(shí)大家都是在vue-cli或者vite創(chuàng)建的vue項(xiàng)目里面使用的這些響應(yīng)式API,今天小編來和大家聊聊如何在node.js項(xiàng)目中使用vue的響應(yīng)式API吧2024-11-11
node.js基于dgram數(shù)據(jù)報(bào)模塊創(chuàng)建UDP服務(wù)器和客戶端操作示例
這篇文章主要介紹了node.js基于dgram數(shù)據(jù)報(bào)模塊創(chuàng)建UDP服務(wù)器和客戶端操作,結(jié)合實(shí)例形式分析了node.js使用dgram數(shù)據(jù)報(bào)模塊創(chuàng)建UDP服務(wù)器和客戶端,以及進(jìn)行UDP廣播、組播相關(guān)操作技巧,需要的朋友可以參考下2020-02-02
利用express啟動(dòng)一個(gè)server服務(wù)的方法
下面小編就為大家?guī)硪黄胑xpress啟動(dòng)一個(gè)server服務(wù)的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-09-09
node.js中express-session配置項(xiàng)詳解
本篇文章主要介紹了node.js中express-session配置項(xiàng)詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-05-05

