阿裡面試題BIO和NIO數量問題附答案和代碼

来源:https://www.cnblogs.com/yuxiang1/archive/2019/03/15/10539924.html
-Advertisement-
Play Games

一、問題 BIO 和 NIO 作為 Server 端,當建立了 10 個連接時,分別產生多少個線程? 答案: 因為傳統的 IO 也就是 BIO 是同步線程堵塞的,所以每個連接都要分配一個專用線程來處理請求,這樣 10 個連接就會創建 10 個線程去處理。而 NIO 是一種同步非阻塞的 I/O 模型, ...


一、問題

BIO 和 NIO 作為 Server 端,當建立了 10 個連接時,分別產生多少個線程?

答案: 因為傳統的 IO 也就是 BIO 是同步線程堵塞的,所以每個連接都要分配一個專用線程來處理請求,這樣 10 個連接就會創建 10 個線程去處理。而 NIO 是一種同步非阻塞的 I/O 模型,它的核心技術是多路復用,可以使用一個鏈接上的不同通道來處理不同的請求,所以即使有 10 個連接,對於 NIO 來說,開啟 1 個線程就夠了。

二、BIO 代碼實現

publicclassDemoServerextendsThread{
privateServerSocket serverSocket;
publicint getPort(){
return serverSocket.getLocalPort();
}
publicvoid run(){
try{
serverSocket =newServerSocket(0);
while(true){
Socket socket = serverSocket.accept();
RequestHandler requestHandler =newRequestHandler(socket);
requestHandler.start();
}
}catch(IOException e){
e.printStackTrace();
}finally{
if(serverSocket !=null){
try{
serverSocket.close();
}catch(IOException e){
e.printStackTrace();
}
}
}
}
publicstaticvoid main(String[] args)throwsIOException{
DemoServer server =newDemoServer();
server.start();
try(Socket client =newSocket(InetAddress.getLocalHost(), server.getPort())){
BufferedReader bufferedReader =newBufferedReader(newInputStreamReader(client.getInputStream()));
bufferedReader.lines().forEach(s ->System.out.println(s));
}
}
}
// 簡化實現,不做讀取,直接發送字元串
classRequestHandlerextendsThread{
privateSocket socket;
RequestHandler(Socket socket){
this.socket = socket;
}
@Override
publicvoid run(){
try(PrintWriter out =newPrintWriter(socket.getOutputStream());){
out.println("Hello world!");
out.flush();
}catch(Exception e){
e.printStackTrace();
}
}
}

 

 

  • 伺服器端啟動 ServerSocket,埠 0 表示自動綁定一個空閑埠。
  • 調用 accept 方法,阻塞等待客戶端連接。
  • 利用 Socket 模擬了一個簡單的客戶端,只進行連接、讀取、列印。
  • 當連接建立後,啟動一個單獨線程負責回覆客戶端請求。

這樣,一個簡單的 Socket 伺服器就被實現出來了。

 

 

(圖片來源於楊曉峰)

三、NIO 代碼實現

 

publicclassNIOServerextendsThread{
publicvoid run(){
try(Selector selector =Selector.open();
ServerSocketChannel serverSocket =ServerSocketChannel.open();){// 創建 Selector 和 Channel
serverSocket.bind(newInetSocketAddress(InetAddress.getLocalHost(),8888));
serverSocket.configureBlocking(false);
// 註冊到 Selector,並說明關註點
serverSocket.register(selector,SelectionKey.OP_ACCEPT);
while(true){
selector.select();// 阻塞等待就緒的 Channel,這是關鍵點之一
Set<SelectionKey> selectedKeys = selector.selectedKeys();
Iterator<SelectionKey> iter = selectedKeys.iterator();
while(iter.hasNext()){
SelectionKey key = iter.next();
// 生產系統中一般會額外進行就緒狀態檢查
sayHelloWorld((ServerSocketChannel) key.channel());
iter.remove();
}
}
}catch(IOException e){
e.printStackTrace();
}
}
privatevoid sayHelloWorld(ServerSocketChannel server)throwsIOException{
try(SocketChannel client = server.accept();){ client.write(Charset.defaultCharset().encode("Hello world!"));
}
}
// 省略了與前面類似的 main
}

 

 

  • 首先,通過 Selector.open() 創建一個 Selector,作為類似調度員的角色。
  • 然後,創建一個 ServerSocketChannel,並且向 Selector 註冊,通過指定 SelectionKey.OP_ACCEPT,告訴調度員,它關註的是新的連接請求。註意:為什麼我們要明確配置非阻塞模式呢?這是因為阻塞模式下,註冊操作是不允許的,會拋出 IllegalBlockingModeException 異常。
  • Selector 阻塞在 select 操作,當有 Channel 發生接入請求,就會被喚醒。
  • 在 sayHelloWorld 方法中,通過 SocketChannel 和 Buffer 進行數據操作,在本例中是發送了一段字元串。

可以看到,在前面兩個樣例中,IO 都是同步阻塞模式,所以需要多線程以實現多任務處理。而 NIO 則是利用了單線程輪詢事件的機制,通過高效地定位就緒的 Channel,來決定做什麼,僅僅 select 階段是阻塞的,可以有效避免大量客戶端連接時,頻繁線程切換帶來的問題,應用的擴展能力有了非常大的提高。下麵這張圖對這種實現思路進行了形象地說明。

 

 

作者: 王磊的博客

免費Java資料領取,涵蓋了Java、Redis、MongoDB、MySQL、Zookeeper、Spring Cloud、Dubbo/Kafka、Hadoop、Hbase、Flink等高併發分散式、大數據、機器學習等技術。
傳送門:

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 什麼是BOM? bom即browser object model 也就是瀏覽器對象模型,BOM由多個對象組成,其中代表瀏覽器視窗的window對象是BOM的頂層對象,其他對象都是該對象的子對象。 頂層對象:window 子對象:location:location 對象包含有關當前 URL 的信息; ...
  • 概述 Prometheus的主要特點 組件 結構圖 適用場景 不適用場景 安裝node_exporter,系統性能指數收集(收集系統性能情況) 下載文件 解壓並複製node_exporter應用程式到/usr/local/bin 清理下載的文件和解壓的文件夾 添加自啟動服務 添加服務配置文件 寫入配 ...
  • 都說三十而立,但現在三十成了程式員的一道坎,如果近年來你有過求職的經歷,或是你關註過智聯、前程、獵聘等招聘平臺,你會發現平臺上數以萬計的招聘信息都會要求應聘者年齡在35歲以內,對於部分科技類初創企業,公司寧願放寬教育程度的要求。也就是說,如果你的年齡大於30歲,那就不要費神來申請了,因為你很可能在自 ...
  • 一、什麼是面向對象 在用面向對象思想寫代碼之前,先瞭解一下什麼是面向對象? 個人理解: 面向對象:把現實世界里的具體物體或者邏輯世界的邏輯物體,用抽象手段,把這些物體抽象成程式能夠識別的類,使類具備物體的屬性和行為,把物體與物體之間的關聯轉換成類與類之間的關聯,用編程邏輯把這些關聯表示出來設計成程式 ...
  • import java.util.Scanner;public class Main{public static void main(String[] args) {int maxn=1000000+5;int mod=10007;int[] Fibonacci=new int[maxn];Fibo ...
  • Python描述符的使用 前言 作為一位python的使用者,你可能使用python有一段時間了,但是對於python中的描述符卻未必使用過,接下來是對描述符使用的介紹 場景介紹 為了引入描述符的使用,我們先設計一個非常簡單的類: class Product(): def __init__(self ...
  • 加密視頻 在以後的開發項目中,很可能有做線上視頻的,而線上視頻就有個問題,因為線上播放,就很有可能視頻數據被抓包,如果這個線上視頻平臺有付費視頻的話,這樣就會有人做點倒賣視頻的生意了,針對這個問題,目前國內有很多不錯的加密視頻平臺,可以把你平臺的視頻放在他們那裡,然後通過他們的機制進行加密,然後做一... ...
  • 個人筆記,如有疏漏,還請指正。 使用多線程(threading)和多進程(multiprocessing)完成常規的併發需求,在啟動的時候 start、join 等步驟不能省,複雜的需要還要用 1 2 個隊列。 隨著需求越來越複雜,如果沒有良好的設計和抽象這部分的功能層次,代碼量越多調試的難度就越大 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...