同步、異步、阻塞、非阻塞四種調用方式


 在進行網絡編程時,我們常常見到同步(Sync)/異步(Async),阻塞(Block)/非阻塞(Unblock)四種調用方式:

同步/異步主要針對C端: 

同步(Sync)

所謂同步,就是發出一個功能調用時,在沒有得到結果之前,該調用就不返回或繼續執行后續操作。

根據這個定義,Java中所有方法都是同步調用,應為必須要等到結果后才會繼續執行。我們在說同步、異步的時候,一般而言是特指那些需要其他端協作或者需要一定時間完成的任務。

簡單來說,同步就是必須一件一件事做,等前一件做完了才能做下一件事。

例如:B/S模式中的表單提交,具體過程是:客戶端提交請求->等待服務器處理->處理完畢返回,在這個過程中客戶端(瀏覽器)不能做其他事。

異步(Async)

異步與同步相對,當一個異步過程調用發出后,調用者在沒有得到結果之前,就可以繼續執行后續操作。當這個調用完成后,一般通過狀態、通知和回調來通知調用者。對於異步調用,調用的返回並不受調用者控制。

對於通知調用者的三種方式,具體如下:

狀態

即監聽被調用者的狀態(輪詢),調用者需要每隔一定時間檢查一次,效率會很低。

通知

當被調用者執行完成后,發出通知告知調用者,無需消耗太多性能。

回調

與通知類似,當被調用者執行完成后,會調用調用者提供的回調函數。

例如:B/S模式中的ajax請求,具體過程是:客戶端發出ajax請求->服務端處理->處理完畢執行客戶端回調,在客戶端(瀏覽器)發出請求后,仍然可以做其他的事。

總結來說,同步和異步的區別:請求發出后,是否需要等待結果,才能繼續執行其他操作。

阻塞/非阻塞主要針對S端:

阻塞
     阻塞調用是指調用結果返回之前,當前線程會被掛起(線程進入非可執行狀態,在這個狀態下,cpu不會給線程分配時間片,即線程暫停運行)。函數只有在得到結果之后才會返回。

 

     有人也許會把阻塞調用和同步調用等同起來,實際上他是不同的。對於同步調用來說,很多時候當前線程還是激活的,只是從邏輯上當前函數沒有返回而已。 例如,我們在socket中調用recv函數,如果緩沖區中沒有數據,這個函數就會一直等待,直到有數據才返回。而此時,當前線程還會繼續處理各種各樣的消息。

   快遞的例子:比如到你某個時候到A樓一層(假如是內核緩沖區)取快遞,但是你不知道快遞什么時候過來,你又不能干別的事,只能死等着。但你可以睡覺(進程處於休眠狀態),因為你知道快遞把貨送來時一定會給你打個電話(假定一定能叫醒你)。

 

非阻塞
      非阻塞和阻塞的概念相對應,指在不能立刻得到結果之前,該函數不會阻塞當前線程,而會立刻返回。

     還是等快遞的例子:如果用忙輪詢的方法,每隔5分鍾到A樓一層(內核緩沖區)去看快遞來了沒有。如果沒來,立即返回。而快遞來了,就放在A樓一層,等你去取。


對象的阻塞模式和阻塞函數調用
對象是否處於阻塞模式和函數是不是阻塞調用有很強的相關性,但是並不是一一對應的。阻塞對象上可以有非阻塞的調用方式,我們可以通過一定的API去輪詢狀 態,在適當的時候調用阻塞函數,就可以避免阻塞。而對於非阻塞對象,調用特殊的函數也可以進入阻塞調用。函數select就是這樣的一個例子。

 

1. 同步,就是我客戶端(c端調用者)調用一個功能,該功能沒有結束前,我(c端調用者)死等結果。
2. 異步,就是我(c端調用者)調用一個功能,不需要知道該功能結果,該功能有結果后通知我(c端調用者)即回調通知。

同步/異步主要針對C端, 但是跟S端不是完全沒有關系,同步/異步機制必須S端配合才能實現. 同步/異步是由c端自己控制, 但是S端是否阻塞/非阻塞, C端完全不需要關心.


3. 阻塞,      就是調用我(s端被調用者,函數),我(s端被調用者,函數)沒有接收完數據或者沒有得到結果之前,我不會返回。
4. 非阻塞,  就是調用我(s端被調用者,函數),我(s端被調用者,函數)立即返回,通過select通知調用者

 

 

同步IO和異步IO的區別就在於:數據訪問的時候進程是否阻塞!

阻塞IO和非阻塞IO的區別就在於:應用程序的調用是否立即返回!

 
同步和異步都只針對於本機SOCKET而言的。

同步和異步,阻塞和非阻塞,有些混用,其實它們完全不是一回事,而且它們修飾的對象也不相同。
阻塞和非阻塞是指當server端的進程訪問的數據如果尚未就緒,進程是否需要等待,簡單說這相當於函數內部的實現區別,也就是未就緒時是直接返回還是等待就緒;

而同步和異步是指client端訪問數據的機制,同步一般指主動請求並等待I/O操作完畢的方式,當數據就緒后在讀寫的時候必須阻塞(區別就緒與讀寫二個階段,同步的讀寫必須阻塞),異步則指主動請求數據后便可以繼續處理其它任務,隨后等待I/O,操作完畢的通知,這可以使進程在數據讀寫時也不阻塞。(等待"通知")

我可以輸,但我不會認輸


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM