大白話搞懂什麼是同步/非同步/阻塞/非阻塞

深夜裡的程式猿發表於2019-04-17

前言

在最近的一些面試中,跟應聘者聊了比較多關於“同步/非同步,阻塞/非阻塞”相關的話題,發現大家對於這些概念的理解都比較模糊,甚至有的同學會反問“他們不就是同一個東西嗎?”。所以藉著這麼一個機會,我想用一些儘量簡單的例子,儘量簡潔的語言來聊聊自己對於這些概念的看法。

正文

這篇文章想通過一個老王“候車”的案例來解釋這些概念。

同步阻塞

放假了,老王回到了鄉下,由於鄉下的基礎設施比較差,當他在車站候車的時候,只能一直在乾等著,直到公交車的到站。

這時候對於公交車(被呼叫著者)來說,它是“同步“的。老王(呼叫者)被公交車(被呼叫者)“阻塞”在站臺上。

非同步阻塞

放完假了,老王回到了大城市開始上班,同樣在車站候車,一樣在車站乾等著,但是大城市的基礎設施建設得比較好,當公交車到站的時候,會有廣播提示提醒乘客。

那麼這時候對於公交車(被呼叫著者)來說,它是“非同步“的,到站後會通知呼叫者。但是此時老王(呼叫者)還是被公交車(被呼叫者)“阻塞”在站臺上。

同步非阻塞

過年了,老王放假回來了鄉下,又要開始候車了,這時候他變聰明瞭,沒有一直在車站上乾等著,而是去找隔壁的小花敘敘舊。但是又害怕車到站了自己會錯過,就只能隔一段時間過來看看車到了沒。

那麼這時候對於公交車(被呼叫著者)來說,它是“同步“的。但是此時老王(呼叫者)可以在候車的時候去幹其他的的事情,所以他是“非阻塞”的。

非同步非阻塞

改革春風吹滿地,新農村建設正在火熱進行中,此時的鄉下,公交車裡面也安裝了車輛到站的提醒廣播。現在老王在候車的時候,可以安心的跟小花敘舊了,當聽到自己需要乘坐的車輛到站廣播時,才過去車站上車。

這時候對於公交車(被呼叫著者)來說,它是“非同步“的,到站後會廣播提醒,此時老王(呼叫者)可以在候車的時候去幹其他的的事情,所以他是“非阻塞”的

概念總結

從上面的示例中,我們可以明白一件事情,同步非同步,阻塞非阻塞他們針對的物件是不一樣的。對於呼叫者來說是阻塞跟非阻塞,被呼叫者是同步跟非同步。

同步:A呼叫B,此時只有等B有結果了才返回。
非同步: A呼叫B,B立即返回,無須等待。當B處理完之後會通過通知或者回撥函式的方式來告訴A結果。
阻塞:A呼叫B,A會被被掛起,一直在等待B的結果,什麼事都不能幹。
非阻塞:A呼叫B,自己用被掛起等待B的結果,可以去幹其他的事情。

Java中相關概念

在Java中的IO模型有三種,分別是BIO(同步阻塞IO),NIO(同步非阻塞IO),AIO(非同步非阻塞IO)。這時候我們會發現,非同步阻塞的模型是不存在的。

NIO跟AIO的出現解決了很多在BIO使用過程遇到的難題,所以我們在選擇使用何種IO的時候需要根據業務場景來做決定,沒必要一味追求NIO跟AIO,不僅加大了編碼的難度也提高的出錯的概率,技術的出現是為了更好的解決問題。

結語

這篇文章主要是想通過大家熟悉的場景來描述這些概念的含義以及區別,如果想更深入的去鑽研的話,大家可以去查閱Linux IO模型相關資料,Java的IO API也是基於這些基礎模型來封裝的。


推薦閱讀

Java異常處理最佳實踐及陷阱防範
論JVM爆炸的幾種姿勢及自救方法
女朋友也能看懂的Zookeeper分散式鎖原理
解放程式設計師雙手之Supervisor

有收穫的話,就點個贊吧

關注「深夜裡的程式猿」,分享最乾的乾貨

大白話搞懂什麼是同步/非同步/阻塞/非阻塞

相關文章