日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java Socket模擬實現聊天室

瀏覽:23日期:2022-08-09 11:11:18

使用Java Socket模擬實現了一個聊天室,實現了基本的私聊以及群聊。分為服務器端和客戶端,下面我來介紹一下實現的步驟。

服務器端

服務器端是聊天室的核心所在,主要用來處理客戶端的請求,先來看一下服務器端的主方法:

public static void main(String[] args) {try { ExecutorService executorService = Executors.newFixedThreadPool(100);//最多容納100個客戶端聊天 ServerSocket serverSocket = new ServerSocket(6655);//監聽6655號端口 for (int i = 0; i < 100; i++) {Socket client = serverSocket.accept();System.out.println('有新的用戶連接 ' + client.getInetAddress() +client.getPort());executorService.execute(new ExecuteClientThread(client)); } executorService.shutdown(); serverSocket.close();} catch (Exception e) { e.printStackTrace();} }

首先我創建了一個固定大小為100的線程池,這個聊天室的實現是一個服務器線程對應一個客戶端線程的,就是說線程池的大小就是最大的同時聊天的人數。服務器的執行順序是這樣的:

1.監聽端口,等待客戶端連接

2.如果有客戶端連接到監聽的端口,那么通過accept()方法返回該客戶端的Socket,并且在線程池中啟動一個新的服務器線程用來與剛剛連接的客戶端'溝通'。

3.把接收到的客戶端的Socket構造注入新啟動的服務器線程中,這樣服務器線程就可以獲取到客戶端對應的流。

到這里,服務器已經和客戶端連接成功了,我們現在來看一下服務器線程是如何處理客戶端的請求的,先上一段服務器代碼

private static Map<String, Socket> clientMap = new ConcurrentHashMap<>();//存儲所有的用戶信息 static class ExecuteClientThread implements Runnable {private Socket client;//每一個服務器線程對應一個客戶端線程ExecuteClientThread(Socket client) { this.client = client;}......

代碼的第一行,創建了一個ConcurrentHashmap,這個map不是某個線程中的,而是服務器的static屬性,用來存儲所有客戶端的信息。因為客戶端是有姓名,有Socket的,所以采用K-value的模式來存儲,用戶名作為Key。考慮到線程安全的原因,采用了ConcurrentHashmap,保證了線程安全。

接下來就是剛剛構造注入的、連接的客戶端的Socket了,我們可以通過這個Socket獲取到輸入和輸出流。

然后就是服務器的線程執行的run方法了,具體的就直接看代碼把。都有注釋,就不一一解釋了,以下是所有服務器端的代碼

import java.io.IOException;import java.io.PrintStream;import java.net.ServerSocket;import java.net.Socket;import java.util.*;import java.util.concurrent.ConcurrentHashMap;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.regex.Matcher;import java.util.regex.Pattern; public class Main { private static Map<String, Socket> clientMap = new ConcurrentHashMap<>();//存儲所有的用戶信息 static class ExecuteClientThread implements Runnable {private Socket client;//每一個服務器線程對應一個客戶端線程ExecuteClientThread(Socket client) { this.client = client;} @Overridepublic void run() { boolean Flag = true;//防止一個客戶端多次注冊所做的標記位置 try {PrintStream PrintToCilent = new PrintStream(client.getOutputStream());//服務器向用戶輸出一些提示信息 Scanner scanner = new Scanner(client.getInputStream());String str = null;//用戶外部的輸入信息while (true) { if (scanner.hasNext()) {str = scanner.next();//外部的用戶輸出 Pattern pattern = Pattern.compile('r');//排除特殊符號Matcher matcher = pattern.matcher(str);str = matcher.replaceAll(''); if (str.startsWith('userName')) { String userName = str.split(':')[1]; userRegist(userName, client, Flag); Flag = false;}// 群聊流程else if (str.startsWith('G:')) { PrintToCilent.println('已進入群聊模式!'); groupChat(scanner,client);}// 私聊流程else if (str.startsWith('P')) {//模式 String userName = str.split('-')[1]; PrintToCilent.println('已經進入與'+userName+'的私聊'); privateChat(scanner,userName);}// 用戶退出else if (str.contains('byebye')) { String userName = null; for (String getKey:clientMap.keySet()) {if (clientMap.get(getKey).equals(client)) { userName = getKey;} } System.out.println('用戶'+userName+'下線了..'); clientMap.remove(userName);//將此實例從map中移除} }} } catch (IOException e) {e.printStackTrace(); }} private void userRegist(String userName, Socket client, boolean Flag) throws IOException { PrintStream PrintToCilent = new PrintStream(client.getOutputStream());//服務器向用戶輸出一些提示信息 if(Flag) {System.out.println('用戶' + userName + '上線了!'); clientMap.put(userName, client);//把用戶加入儲存mapSystem.out.println('當前群聊人數為' + (clientMap.size()) + '人');PrintToCilent.println('注冊成功!'); }else {PrintToCilent.println('警告:一個客戶端只能注冊一個用戶!'); }} private void groupChat(Scanner scanner,Socket client) throws IOException { // 取出clientMap中所有客戶端Socket,然后遍歷一遍 // 分別取得每個Socket的輸出流向每個客戶端輸出 PrintStream PrintToClient = new PrintStream(client.getOutputStream());//在群聊的時候服務器向客戶端發送數據 boolean ExitFlag = false; Set<Map.Entry<String, Socket>> entrySet = clientMap.entrySet(); String userName = null; for (Map.Entry<String, Socket> socketEntry : entrySet) {//獲得:是哪個用戶說的話if (socketEntry.getValue() == client) { userName = socketEntry.getKey();//發出信息的用戶} } String msg = null; while (true) {if (scanner.hasNext()) { msg = scanner.next(); if('exit'.equals(msg)){//如果用戶退出了for(Map.Entry<String,Socket> stringSocketEntry : entrySet){ new PrintStream(stringSocketEntry.getValue().getOutputStream(),true).println('用戶'+userName+'剛剛退出了群聊!!');//給所有人發退出群聊的消息}return; } for (Map.Entry<String, Socket> stringSocketEntry : entrySet) {//遍歷用戶的map,獲取所有用戶的Sockettry { Socket socket = stringSocketEntry.getValue(); PrintStream ps = new PrintStream(socket.getOutputStream(), true); ps.println('群聊:用戶' + userName + '說: ' + msg);//給每個用戶發消息} catch (IOException e) { e.printStackTrace();} } } } }private void privateChat(Scanner scanner, String privatepeopleName) throws IOException { Socket privateUser = clientMap.get(privatepeopleName); PrintStream ps = new PrintStream(privateUser.getOutputStream());//拿到私聊對象的輸出流 PrintStream PrintToClient = new PrintStream(client.getOutputStream());//拿到當前客戶端的輸出流 String Message = null; String MyName = null; Set<Map.Entry<String,Socket>> set = clientMap.entrySet(); for(Map.Entry<String,Socket> value : set){if(value.getValue() == client){ MyName = value.getKey(); break;} } while (true) {if(scanner.hasNext()) { Message = scanner.next(); if ('exit'.equals(Message)){//如果用戶輸入了退出PrintToClient.println('已退出和'+privatepeopleName+'的私聊');ps.println('對方已經退出了私聊');break; } ps.println(MyName+'說'+Message);//如果用戶沒有退出,向私聊對象發送消息} } } } public static void main(String[] args) {try { ExecutorService executorService = Executors.newFixedThreadPool(100);//最多容納100個客戶端聊天 ServerSocket serverSocket = new ServerSocket(6655); for (int i = 0; i < 100; i++) {Socket client = serverSocket.accept();System.out.println('有新的用戶連接 ' + client.getInetAddress() +client.getPort());executorService.execute(new ExecuteClientThread(client)); } executorService.shutdown(); serverSocket.close();} catch (Exception e) { e.printStackTrace();} }}

然后是客戶端的代碼,客戶端的代碼比較簡單:分為兩個線程,一個線程用于接收服務器的數據,一個線程用于向服務器發送數據。我就直接上代碼了,里面有注釋的。

import java.io.IOException;import java.io.PrintStream;import java.net.Socket;import java.util.Scanner; class ExcuteServerInPut implements Runnable{//接收服務器的數據 private Socket ToServer; ExcuteServerInPut(Socket ToServer){this.ToServer = ToServer; } @Override public void run() {try { Scanner scanner = new Scanner(ToServer.getInputStream()); while (scanner.hasNext()){System.out.println(scanner.nextLine()); } scanner.close(); ToServer.close();} catch (IOException e) { e.printStackTrace();} }} class ExcuteServerOutPut implements Runnable{//向服務器發送數據 private Socket Socket; ExcuteServerOutPut(Socket Socket){this.Socket = Socket; } @Override public void run() {try { PrintStream printStream = new PrintStream(Socket.getOutputStream()); Scanner scanner = new Scanner(System.in); scanner.useDelimiter('n'); System.out.println('*****************************************'); System.out.println('***用戶注冊:useerName:同戶名(僅限一次)***'); System.out.println('***進入群聊:G: 退出群聊:exit***'); System.out.println('***私聊:P-用戶名 退出私聊:exit***'); System.out.println('***********退出聊天室:byebye*************'); while (true){if(scanner.hasNext()) { String string = scanner.next(); printStream.println(string); if ('byebye'.equals(string)) {System.out.println('退出!');printStream.close();scanner.close();break; }} } Socket.close();} catch (IOException e) { e.printStackTrace();} }} public class Main { public static void main(String[] args) throws IOException {Socket socket = new Socket('127.0.0.1', 6655);ExcuteServerInPut excuteServerInPut = new ExcuteServerInPut(socket);ExcuteServerOutPut excuteServerOutPut = new ExcuteServerOutPut(socket);new Thread(excuteServerInPut).start();new Thread(excuteServerOutPut).start();}}

后續我會做一些改進,希望可以對大家有所幫助

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
黑丝美女一区二区| 国产精品一区二区三区www| 中文无码久久精品| 亚洲一区二区日韩| 欧美在线资源| 日本不卡免费高清视频在线| 男女男精品视频网| 91精品蜜臀一区二区三区在线| 国产三级精品三级在线观看国产| 精品三级久久| 日本电影久久久| 精品国产a一区二区三区v免费| 日韩国产一二三区| 久久av中文| 99久久九九| 亚洲精品三级| 欧美精品不卡| 亚洲天堂黄色| 日日夜夜免费精品| 国产91在线精品| 亚洲精品一区二区在线看| 亚洲丝袜啪啪| 精品精品久久| 在线综合视频| 精品美女在线视频| 亚洲在线成人| 欧美色综合网| 久久精品二区三区| 日韩av午夜在线观看| 日韩欧美二区| 欧美一区影院| 在线亚洲观看| 国产精品精品国产一区二区| 欧美在线网站| 国产成人精品一区二区三区视频 | 久久中文欧美| 一本一本久久| 中文在线免费视频| 日韩成人在线看| 亚洲综合在线电影| 国产精品最新| 免费成人性网站| 99热精品久久| 国产成人77亚洲精品www| 中文字幕免费精品| 91精品电影| 秋霞国产精品| 国产成人免费av一区二区午夜| 亚洲精品一级| 日韩专区一卡二卡| 伊人久久亚洲美女图片| 精品一区不卡| 国产精品xxx| 国产欧美激情| 亚洲五月综合| 中日韩男男gay无套| 在线日韩av| 成人在线超碰| 精品视频91| 国产精品99久久免费观看| 涩涩涩久久久成人精品| 黄色在线一区| 最新亚洲激情| 视频一区视频二区中文字幕| 91精品福利| 国产精品美女久久久| 免费精品国产的网站免费观看| 快播电影网址老女人久久| a日韩av网址| 精品三级av| 国产激情在线播放| 日韩av首页| 午夜久久99| 亚洲午夜国产成人| 在线精品国产亚洲| 日本午夜精品视频在线观看| 日韩欧美另类中文字幕| 欧美日韩a区| 国产一区二区三区网| 亚洲播播91| 欧美精品羞羞答答| 视频一区二区不卡| 国产精品久久免费视频| 麻豆成全视频免费观看在线看| 精品国产亚洲日本| 亚洲天堂久久| 日韩欧美激情电影| 国产精品mv在线观看| 日韩精品久久久久久久电影99爱| 久久精品国语| 日本成人中文字幕在线视频| 国产福利一区二区三区在线播放| 日韩在线欧美| 自拍自偷一区二区三区| 91精品在线观看国产| 国产美女撒尿一区二区| 中文字幕在线视频久| 国产视频亚洲| 国产成人77亚洲精品www| 亚洲激情欧美| 麻豆成人av在线| 亚洲高清av| 美女久久久久久| 免费观看在线综合色| 中文字幕在线看片| 日韩激情综合| 国产综合色产| 日韩成人精品一区| 日本va欧美va精品| 1000部精品久久久久久久久| 欧美亚洲专区| 美女精品在线观看| 欧美成a人国产精品高清乱码在线观看片在线观看久 | 欧美在线资源| 国产一区2区| 欧美日韩xxxx| 免费国产亚洲视频| 久久久久91| 91亚洲国产| 日韩欧美中文字幕一区二区三区 | 久久精品99国产精品日本| 国产亚洲亚洲| 久久久国产亚洲精品| 国产精品观看| 日本不卡在线视频| 亚洲成a人片| 欧美日韩1区2区3区| 欧美伊人久久| 青青草91久久久久久久久| 国产婷婷精品| 亚洲欧美日韩专区| 国产精品欧美在线观看| 国产精品草草| 国产成人久久精品麻豆二区| 精品久久影院| 日韩欧美自拍| 亚洲午夜一级| 日韩中文字幕区一区有砖一区 | 日本午夜精品一区二区三区电影 | 99久久夜色精品国产亚洲狼| 久久久夜精品| 国产视频一区免费看| 亚洲最大av| 欧美日韩精品一区二区三区视频 | 婷婷综合网站| 亚洲一区av| 国产精品www.| 99久久精品国产亚洲精品| 国产精品普通话对白| 日韩精品一页| 美女视频一区在线观看| 国产精品久久久久av电视剧| 亚洲免费成人| 国产亚洲观看| 韩国精品主播一区二区在线观看| 精品在线91| 国产日韩欧美| 91精品1区| 久久久久亚洲精品中文字幕| 日韩中文在线播放| 亚洲视频国产| 日韩av片子| 偷拍亚洲精品| 欧美色图一区| 久久av网站| 亚洲视频国产精品| av综合电影网站| 日本视频中文字幕一区二区三区| 成人精品天堂一区二区三区| 四虎精品一区二区免费| 日韩大片在线播放| 欧美亚洲自偷自偷| 欧美日韩日本国产亚洲在线| 麻豆国产欧美日韩综合精品二区| 亚洲啊v在线| 日韩av在线播放中文字幕| 99久久精品费精品国产| 美女视频黄久久| 日本99精品| 免费久久99精品国产| 亚洲第一精品影视| 国产在线看片免费视频在线观看| 日本综合精品一区| 久久夜色精品| 亚洲大全视频| 久久一区二区三区喷水| 国产欧美一区二区三区精品酒店| 国产欧美日韩免费观看| 综合一区二区三区| 午夜久久tv| 五月天久久网站| 国产综合激情| 亚洲视频播放| 中文无码久久精品| 日韩一区二区三区四区五区 | 精品久久91| 国产在线一区不卡| 国产精品成久久久久| 黄色在线网站噜噜噜| 日韩精品永久网址|