非阻塞套接字編程詳解

阻塞與非阻塞套接字對比

傳統阻塞式套接字編程使用ServerSocketSocket類時,關鍵方法如connect()accept()read()write()都會導致調用線程阻塞,直到操作完成。這種模式存在兩個主要問題:

  1. 客戶端線程在等待數據時會被完全阻塞
  2. 服務端需要為每個客戶端連接創建獨立線程,資源消耗大

核心類對比

阻塞式通信類非阻塞式通信類說明
ServerSocketServerSocketChannel底層仍使用ServerSocket
SocketSocketChannel底層仍使用Socket
InputStream/Output無直接對應類通過SocketChannel進行讀寫
無對應類Selector事件選擇器核心組件
無對應類SelectionKey表示通道注冊的事件類型

非阻塞機制原理

非阻塞套接字通過三個核心組件協同工作:

// 獲取選擇器實例
Selector selector = Selector.open();// 創建非阻塞服務端通道
ServerSocketChannel ssChannel = ServerSocketChannel.open();
ssChannel.configureBlocking(false);  // 必須設置為非阻塞模式
ssChannel.bind(new InetSocketAddress("localhost", 19000));// 注冊ACCEPT事件
ssChannel.register(selector, SelectionKey.OP_ACCEPT);

事件類型與處理

選擇器支持四種事件類型,對應SelectionKey中的常量:

  1. OP_CONNECT - 客戶端連接就緒
  2. OP_ACCEPT - 服務端接受新連接
  3. OP_READ - 數據可讀
  4. OP_WRITE - 數據可寫

典型的事件處理循環如下:

while(true) {int readyCount = selector.select();  // 阻塞直到有事件發生if(readyCount <= 0) continue;Set readyKeys = selector.selectedKeys();Iterator iter = readyKeys.iterator();while(iter.hasNext()) {SelectionKey key = iter.next();iter.remove();if(key.isAcceptable()) {// 處理新連接ServerSocketChannel ssChannel = (ServerSocketChannel)key.channel();SocketChannel clientChannel = ssChannel.accept();clientChannel.configureBlocking(false);clientChannel.register(selector, SelectionKey.OP_READ);}else if(key.isReadable()) {// 讀取數據SocketChannel channel = (SocketChannel)key.channel();ByteBuffer buffer = ByteBuffer.allocate(1024);channel.read(buffer);// ...處理數據...}}
}

性能優勢體現

類比快餐店運營模式:

  • 傳統阻塞模式:每個顧客(客戶端)需要專屬服務員(線程),資源利用率低
  • 非阻塞模式:前臺(Selector)統一接待,廚房(工作線程)并行處理,實現:
    • 單線程處理多連接
    • 資源按需分配
    • 無空閑線程等待

客戶端實現要點

客戶端同樣需要遵循非阻塞模式:

SocketChannel clientChannel = SocketChannel.open();
clientChannel.configureBlocking(false);
clientChannel.connect(new InetSocketAddress("localhost", 19000));// 注冊連接、讀寫事件
clientChannel.register(selector, SelectionKey.OP_CONNECT | SelectionKey.OP_READ | SelectionKey.OP_WRITE);// 處理連接完成事件
if(key.isConnectable()) {while(clientChannel.isConnectionPending()) {clientChannel.finishConnect();  // 完成非阻塞連接}
}

注意事項

  1. 緩沖區管理:必須配合ByteBuffer進行數據讀寫
  2. 字符編碼:需顯式處理字符集編解碼
  3. 事件去重:處理完SelectionKey后需從ready集合移除
  4. 資源釋放:異常時需調用key.cancel()取消注冊

這種模式雖然提高了吞吐量,但也帶來了編程復雜度,適合高并發但單連接數據處理量不大的場景。

核心組件與工作原理

Selector調度機制

Selector作為非阻塞I/O的核心調度中心,通過select()方法監控所有注冊通道的I/O事件狀態。當至少一個通道準備好進行注冊的操作時,select()會返回就緒通道的數量,典型的事件處理循環結構如下:

while (true) {int readyChannels = selector.select(); // 阻塞直到有事件就緒if (readyChannels <= 0) continue;Set readyKeys = selector.selectedKeys();Iterator keyIterator = readyKeys.iterator();while (keyIterator.hasNext()) {SelectionKey key = keyIterator.next();keyIterator.remove(); // 必須顯式移除已處理的keyif (key.isAcceptable()) {handleAccept(key);} else if (key.isReadable()) {handleRead(key);}}
}

四種核心操作類型

通道可注冊的事件類型通過SelectionKey常量定義:

操作類型適用場景檢測方法
OP_ACCEPT服務端接受新連接isAcceptable()
OP_CONNECT客戶端建立連接isConnectable()
OP_READ通道數據可讀isReadable()
OP_WRITE通道可寫入數據isWritable()

組合注冊示例:

// 客戶端通道注冊連接、讀寫事件
channel.register(selector, SelectionKey.OP_CONNECT | SelectionKey.OP_READ |SelectionKey.OP_WRITE);

SelectionKey工作機制

每個注冊通道對應一個SelectionKey,包含三個重要屬性:

  1. interest集合:通道關注的事件類型
  2. ready集合:當前就緒的事件類型
  3. 附加對象:可通過attach()綁定業務對象

關鍵方法:

// 獲取關聯通道
SelectableChannel channel = key.channel(

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/907712.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/907712.shtml
英文地址,請注明出處:http://en.pswp.cn/news/907712.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

電子電路:初步認識CMOS技術

CMOS&#xff08;Complementary Metal-Oxide-Semiconductor&#xff0c;互補金屬氧化物半導體&#xff09;是一種半導體技術&#xff0c;廣泛應用于集成電路&#xff08;IC&#xff09;的設計和制造中。以下是關于CMOS的詳細說明&#xff1a; 1. 基本概念 技術原理&#xff1a…

【11408學習記錄】考研英語寫作提分秘籍:2013真題邀請信精講+萬能模板套用技巧

邀請信 英語寫作2013年考研英語&#xff08;一&#xff09;真題小作文題目分析寫作思路第一段&#xff1a;第二段&#xff1a;錦囊妙句1&#xff1a;錦囊妙句2&#xff1a;錦囊妙句3&#xff1a;錦囊妙句5&#xff1a;錦囊妙句6&#xff1a;錦囊妙句9&#xff1a;錦囊妙句14&am…

Java 注解與反射(超詳細!!!)

Java 注解與反射&#xff08;超詳細&#xff01;&#xff01;&#xff01;&#xff09; 文章目錄 Java 注解與反射&#xff08;超詳細&#xff01;&#xff01;&#xff01;&#xff09;1.注解1.1內置注解1.1.1 SuppressWarnings注解用法 1.2 元注解1.3自定義注解 2.反射2.1 反…

用nz-tabel寫一個合并表格

用nz-tabel寫一個合并表格 <nz-table #basicTable [nzData]"tableSearchStatus.dataList" nzBordered><thead><tr><th>班級</th><th>姓名</th><th>年齡</th><th>電話</th></tr></thead&…

第6章 放大電路的反饋

本章基本要求 會判&#xff1a;判斷電路中有無反饋及反饋的性質 會算&#xff1a;估算深度負反饋條件下的放大倍數 會引&#xff1a;根據需求引入合適的反饋 會判振消振&#xff1a;判斷電路是否能穩定工作&#xff0c;會消除自激振蕩。 6.1 反饋的概念及判斷 一、反饋的…

ansible template 文件中如果包含{{}} 等非ansible 變量處理

在 Ansible 模板中&#xff0c;如果你的 Python 腳本里有大量 {}、f""、或者其他 Jinja 會誤解析的語法&#xff0c;就需要用 {% raw %}…{% endraw %} 把它們包起來&#xff0c;只在需要替換變量的那一行單獨“放行”。例如&#xff1a; {% raw %} #!/usr/bin/env …

STM32G4 電機外設篇(一) GPIO+UART

目錄 一、STM32G4 電機外設篇&#xff08;一&#xff09; GPIOUART1 GPIO1.1 STM32CUBEMX 配置以及Keil代碼1.2 代碼和實驗現象 2 UART2.1 STM32CUBEMX 配置以及Keil代碼2.2 代碼和實驗現象 附學習參考網址歡迎大家有問題評論交流 (* ^ ω ^) 一、STM32G4 電機外設篇&#xff0…

Kotlin 中集合遍歷有哪幾種方式?

1 for-in 循環&#xff08;最常用&#xff09; val list listOf("A", "B", "C") for (item in list) {print("$item ") }// A B C 2 forEach 高階函數 val list listOf("A", "B", "C") list.forEac…

尚硅谷redis7 99 springboot整合redis之連接集群

6381宕機&#xff0c;手動shutdown后在redis中&#xff0c;634自動上位變成master結點。 但是在springboot中卻沒有動態感知道redisCluster的最新集群消息&#xff0c;所以找不到我們要檢索的數據。原因是&#xff1a;SpringBoot 2.X版本,Redis默認的連接池采用 Lettuce&#…

AI 的早期萌芽?用 Swift 演繹約翰·康威的「生命游戲」

文章目錄 摘要描述題解答案題解代碼分析示例測試及結果時間復雜度空間復雜度總結 摘要 你有沒有想過&#xff0c;能不能通過簡單的規則模擬出生與死亡&#xff1f;「生命游戲」正是這樣一種充滿魅力的數學模擬系統。這篇文章我們來聊聊它的規則到底有多神奇&#xff0c;并用 S…

web ui自動化工具playwright

playwright是微軟開源的一款web ui自動化工具&#xff0c;該工具有很多亮點&#xff0c;解決以前困擾web UI自動化測試的很多難點。這篇博客將介紹playwright主要特點。 playwright支持錄制減少了編寫成本 如果要使用playwright的錄制功能&#xff0c;有兩種途徑&#xff0c;途…

移動安全Android——客戶端靜態安全

一、反編譯保護 測試工具 Jadx GitHub - skylot/jadx: Dex to Java decompiler PKID [下載]PKID-APP查殼工具-Android安全-看雪-安全社區|安全招聘|kanxue.com 測試流程 &#xff08;1&#xff09;通過Jadx對客戶端APK文件進行反編譯&#xff0c;觀察是否進行代碼混淆 &…

04-redis-分布式鎖-edisson

1 基本概念 百度百科&#xff1a;控制分布式系統之間同步訪問共享資源方式。 在分布式系統中&#xff0c;常常需要協調他們的動作。如果不同的系統或是同一個系統的不同主機之間共享了一個或一組資源&#xff0c;那么訪問這些資源的時候&#xff0c;往往需要互斥來防止…

cf每日刷題

目錄 String&#xff08;800&#xff09; Skibidus and Amogu&#xff08;800&#xff09; Apples in Boxes&#xff08;1100&#xff09; String&#xff08;800&#xff09; https://codeforces.com/problemset/problem/2062/A #include <iostream> #include <…

AWS WebRTC:獲取ICE服務地址(part 1)

建立WebRTC連接的第二步是獲取ICE服務地址。 ICE全稱&#xff1a;Interactive Connectivity Establishment&#xff0c;建立互動連接。 ICE 服務地址&#xff0c;主要是 TURN 和 STUN 服務器的地址&#xff0c;用于 WebRTC 在 NAT 網絡環境中協商建立連接。 上代碼&#xff…

Python興趣匹配算法:從理論到實戰的進階指南

目錄 一、興趣匹配算法的技術棧解析 1. 基礎特征匹配階段 2. 向量空間模型階段 3. 深度學習階段 二、工程化實踐關鍵技術 1. 特征工程體系 2. 相似度計算優化 三、典型應用場景實現 1. 社交好友推薦系統 2. 電商商品推薦系統 四、性能優化與挑戰應對 1. 計算性能優…

【C語言】講解 程序分配的區域(新手)

目錄 代碼區 數據區 堆區 棧區 常量區 重點比較一下堆區與 棧區 總結&#xff1a; 前言&#xff1a; C語言程序的內存分配區域是理解其運行機制的重要部分。根據提供的多條證據&#xff0c;我們可以總結出C語言程序在運行時主要涉及以下五個關鍵內存區域&#xff1a; 代…

Go語言之接口與多態 -《Go語言實戰指南》

接口是 Go 語言實現 多態 的核心機制。本章將幫助你理解接口的設計哲學、動態行為&#xff0c;以及它如何讓 Go 實現面向接口編程的能力。 一、什么是接口&#xff1f; 接口是一組方法簽名的集合&#xff0c;任何類型只要實現了接口中聲明的所有方法&#xff0c;就被視為實現了…

JSR 303(即 Bean Validation)是一個通過??注解在 Java Bean 上定義和執行驗證規則??的規范

&#x1f6e0;? 一、JSR 303是什么&#xff1f; JSR 303&#xff08;Java Specification Requests 303&#xff09;是Java EE 6的子規范&#xff0c;全稱??Bean Validation??。它通過注解方式對JavaBean的屬性值進行標準化校驗&#xff0c;例如檢查非空、長度、格式等規則…

【圖像處理入門】3. 幾何變換基礎:從平移旋轉到插值魔法

摘要 掌握圖像的幾何變換相當于學會「圖像的空間魔法」。本文將帶你理解平移/旋轉/縮放的數學原理&#xff0c;掌握OpenCV中warpAffine和getAffineTransform的核心用法&#xff0c;對比最近鄰、雙線性等插值算法的優劣。通過圖像翻轉、鏡像、透視變換實戰&#xff0c;學會用變…