Java?NIO實(shí)戰(zhàn)之多人聊天室
本文實(shí)例為大家分享了Java NIO實(shí)戰(zhàn)之多人聊天室的具體代碼,供大家參考,具體內(nèi)容如下
NIO服務(wù)端
public class NioServer { /** * 啟動(dòng) */ public void start() throws IOException { /** * 1. 創(chuàng)建Selector */ Selector selector = Selector.open(); /** * 2. 通過(guò)ServerSocketChannel創(chuàng)建channel通道 */ ServerSocketChannel serverSocketChannel = ServerSocketChannel.open(); /** * 3. 為channel通道綁定監(jiān)聽(tīng)端口 */ serverSocketChannel.bind(new InetSocketAddress(8000)); /** * 4. **設(shè)置channel為非阻塞模式** */ serverSocketChannel.configureBlocking(false); /** * 5. 將channel注冊(cè)到selector上,監(jiān)聽(tīng)連接事件 */ serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT); System.out.println("服務(wù)器啟動(dòng)成功!"); /** * 6. 循環(huán)等待新接入的連接 */ for (;;) { // while(true) c for;; /** * TODO 獲取可用channel數(shù)量 */ int readyChannels = selector.select(); /** * TODO 為什么要這樣!??? */ if (readyChannels == 0) continue; /** * 獲取可用channel的集合 */ Set<SelectionKey> selectionKeys = selector.selectedKeys(); Iterator iterator = selectionKeys.iterator(); while (iterator.hasNext()) { /** * selectionKey實(shí)例 */ SelectionKey selectionKey = (SelectionKey) iterator.next(); /** * **移除Set中的當(dāng)前selectionKey** */ iterator.remove(); /** * 7. 根據(jù)就緒狀態(tài),調(diào)用對(duì)應(yīng)方法處理業(yè)務(wù)邏輯 */ /** * 如果是 接入事件 */ if (selectionKey.isAcceptable()) { acceptHandler(serverSocketChannel, selector); } /** * 如果是 可讀事件 */ if (selectionKey.isReadable()) { readHandler(selectionKey, selector); } } } } /** * 接入事件處理器 */ private void acceptHandler(ServerSocketChannel serverSocketChannel, Selector selector) throws IOException { /** * 如果要是接入事件,創(chuàng)建socketChannel */ SocketChannel socketChannel = serverSocketChannel.accept(); /** * 將socketChannel設(shè)置為非阻塞工作模式 */ socketChannel.configureBlocking(false); /** * 將channel注冊(cè)到selector上,監(jiān)聽(tīng) 可讀事件 */ socketChannel.register(selector, SelectionKey.OP_READ); /** * 回復(fù)客戶端提示信息 */ socketChannel.write(Charset.forName("UTF-8") .encode("你與聊天室里其他人都不是朋友關(guān)系,請(qǐng)注意隱私安全")); } /** * 可讀事件處理器 */ private void readHandler(SelectionKey selectionKey, Selector selector) throws IOException { /** * 要從 selectionKey 中獲取到已經(jīng)就緒的channel */ SocketChannel socketChannel = (SocketChannel) selectionKey.channel(); /** * 創(chuàng)建buffer */ ByteBuffer byteBuffer = ByteBuffer.allocate(1024); /** * 循環(huán)讀取客戶端請(qǐng)求信息 */ String request = ""; while (socketChannel.read(byteBuffer) > 0) { /** * 切換buffer為讀模式 */ byteBuffer.flip(); /** * 讀取buffer中的內(nèi)容 */ request += Charset.forName("UTF-8").decode(byteBuffer); } /** * 將channel再次注冊(cè)到selector上,監(jiān)聽(tīng)他的可讀事件 */ socketChannel.register(selector, SelectionKey.OP_READ); /** * 將客戶端發(fā)送的請(qǐng)求信息 廣播給其他客戶端 */ if (request.length() > 0) { // 廣播給其他客戶端 broadCast(selector, socketChannel, request); } } /** * 廣播給其他客戶端 */ private void broadCast(Selector selector, SocketChannel sourceChannel, String request) { /** * 獲取到所有已接入的客戶端channel */ Set<SelectionKey> selectionKeySet = selector.keys(); /** * 循環(huán)向所有channel廣播信息 */ selectionKeySet.forEach(selectionKey -> { Channel targetChannel = selectionKey.channel(); // 剔除發(fā)消息的客戶端 if (targetChannel instanceof SocketChannel && targetChannel != sourceChannel) { try { // 將信息發(fā)送到targetChannel客戶端 ((SocketChannel) targetChannel).write( Charset.forName("UTF-8").encode(request)); } catch (IOException e) { e.printStackTrace(); } } }); } /** * 主方法 * @param args */ public static void main(String[] args) throws IOException { new NioServer().start(); } }
NIO客戶端
public class NioClient { /** * 啟動(dòng) */ public void start(String nickname) throws IOException { /** * 連接服務(wù)器端 */ SocketChannel socketChannel = SocketChannel.open( new InetSocketAddress("127.0.0.1", 8000)); /** * 接收服務(wù)器端響應(yīng) */ // 新開(kāi)線程,專(zhuān)門(mén)負(fù)責(zé)來(lái)接收服務(wù)器端的響應(yīng)數(shù)據(jù) // selector , socketChannel , 注冊(cè) Selector selector = Selector.open(); socketChannel.configureBlocking(false); socketChannel.register(selector, SelectionKey.OP_READ); new Thread(new NioClientHandler(selector)).start(); /** * 向服務(wù)器端發(fā)送數(shù)據(jù) */ Scanner scanner = new Scanner(System.in); while (scanner.hasNextLine()) { String request = scanner.nextLine(); if (request != null && request.length() > 0) { socketChannel.write( Charset.forName("UTF-8") .encode(nickname + " : " + request)); } } } public static void main(String[] args) throws IOException { // new NioClient().start(); } }
客戶端線程,處理服務(wù)器端響應(yīng)的的消息
public class NioClientHandler implements Runnable { private Selector selector; public NioClientHandler(Selector selector) { this.selector = selector; } @Override public void run() { try { for (;;) { int readyChannels = selector.select(); if (readyChannels == 0) continue; /** * 獲取可用channel的集合 */ Set<SelectionKey> selectionKeys = selector.selectedKeys(); Iterator iterator = selectionKeys.iterator(); while (iterator.hasNext()) { /** * selectionKey實(shí)例 */ SelectionKey selectionKey = (SelectionKey) iterator.next(); /** * **移除Set中的當(dāng)前selectionKey** */ iterator.remove(); /** * 7. 根據(jù)就緒狀態(tài),調(diào)用對(duì)應(yīng)方法處理業(yè)務(wù)邏輯 */ /** * 如果是 可讀事件 */ if (selectionKey.isReadable()) { readHandler(selectionKey, selector); } } } } catch (IOException e) { e.printStackTrace(); } } /** * 可讀事件處理器 */ private void readHandler(SelectionKey selectionKey, Selector selector) throws IOException { /** * 要從 selectionKey 中獲取到已經(jīng)就緒的channel */ SocketChannel socketChannel = (SocketChannel) selectionKey.channel(); /** * 創(chuàng)建buffer */ ByteBuffer byteBuffer = ByteBuffer.allocate(1024); /** * 循環(huán)讀取服務(wù)器端響應(yīng)信息 */ String response = ""; while (socketChannel.read(byteBuffer) > 0) { /** * 切換buffer為讀模式 */ byteBuffer.flip(); /** * 讀取buffer中的內(nèi)容 */ response += Charset.forName("UTF-8").decode(byteBuffer); } /** * 將channel再次注冊(cè)到selector上,監(jiān)聽(tīng)他的可讀事件 */ socketChannel.register(selector, SelectionKey.OP_READ); /** * 將服務(wù)器端響應(yīng)信息打印到本地 */ if (response.length() > 0) { System.out.println(response); } } }
我們定義三個(gè)客戶端,模擬三個(gè)用戶在聊天室發(fā)送消息
public class AClient { public static void main(String[] args) throws IOException { new NioClient().start("AClient"); } } public class BClient { public static void main(String[] args) throws IOException { new NioClient().start("BClient"); } } public class CClient { public static void main(String[] args) throws IOException { new NioClient().start("CClient"); } }
NIO 聊天室到此結(jié)束
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
面向切面的Spring通過(guò)切點(diǎn)來(lái)選擇連接點(diǎn)實(shí)例詳解
這篇文章主要為大家介紹了面向切面的Spring通過(guò)切點(diǎn)來(lái)選擇連接點(diǎn)實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10SpringBoot+WebSocket+Netty實(shí)現(xiàn)消息推送的示例代碼
這篇文章主要介紹了SpringBoot+WebSocket+Netty實(shí)現(xiàn)消息推送的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04Java實(shí)現(xiàn)讀取Jar文件屬性的方法詳解
這篇文章主要為大家詳細(xì)介紹了如何利用Java語(yǔ)言實(shí)現(xiàn)讀取Jar文件屬性的功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-08-08Springcloud-nacos實(shí)現(xiàn)配置和注冊(cè)中心的方法
這篇文章主要介紹了Springcloud-nacos實(shí)現(xiàn)配置和注冊(cè)中心的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07spring的pointcut正則表達(dá)式的實(shí)現(xiàn)
本文主要介紹了spring的pointcut正則表達(dá)式的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08SpringBoot發(fā)送郵件功能 驗(yàn)證碼5分鐘過(guò)期
這篇文章主要為大家詳細(xì)介紹了SpringBoot發(fā)送郵件功能,驗(yàn)證碼5分鐘過(guò)期,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03