java Socket實(shí)現(xiàn)多人群聊與私聊功能
本文實(shí)例為大家分享了java Socket實(shí)現(xiàn)多人群聊與私聊的具體代碼,供大家參考,具體內(nèi)容如下
關(guān)于Socket套接字的一些基本知識(shí)與認(rèn)識(shí)可以參見(jiàn)上一篇或自行查閱。
ServerSocket和Socket實(shí)現(xiàn)群聊與私聊涉及到多線程編程,實(shí)現(xiàn)過(guò)程的重點(diǎn)是利用Socket通信的原理,即不斷的在服務(wù)端和客戶端創(chuàng)建輸入輸出流來(lái)相互傳遞、交換數(shù)據(jù)等以達(dá)到通信的目的。具體實(shí)現(xiàn)如下:
服務(wù)端:
import java.io.*; import java.net.*; import java.util.HashMap; import java.util.Map; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.ThreadPoolExecutor; public class TCPServer { private ServerSocket serverSocket; /** * 創(chuàng)建線程池來(lái)管理客戶端的連接線程 * 避免系統(tǒng)資源過(guò)度浪費(fèi) */ private ExecutorService exec; // 存放客戶端之間私聊的信息 private Map<String,PrintWriter> storeInfo; public TCPServer() { try { serverSocket = new ServerSocket(6789); storeInfo = new HashMap<String, PrintWriter>(); exec = Executors.newCachedThreadPool(); } catch (Exception e) { e.printStackTrace(); } } // 將客戶端的信息以Map形式存入集合中 private void putIn(String key,PrintWriter value) { synchronized(this) { storeInfo.put(key, value); } } // 將給定的輸出流從共享集合中刪除 private synchronized void remove(String key) { storeInfo.remove(key); System.out.println("當(dāng)前在線人數(shù)為:"+ storeInfo.size()); } // 將給定的消息轉(zhuǎn)發(fā)給所有客戶端 private synchronized void sendToAll(String message) { for(PrintWriter out: storeInfo.values()) { out.println(message); } } // 將給定的消息轉(zhuǎn)發(fā)給私聊的客戶端 private synchronized void sendToSomeone(String name,String message) { PrintWriter pw = storeInfo.get(name); //將對(duì)應(yīng)客戶端的聊天信息取出作為私聊內(nèi)容發(fā)送出去 if(pw != null) pw.println(message); } public void start() { try { while(true) { System.out.println("等待客戶端連接... ... "); Socket socket = serverSocket.accept(); // 獲取客戶端的ip地址 InetAddress address = socket.getInetAddress(); System.out.println("客戶端:“" + address.getHostAddress() + "”連接成功! "); /** * 啟動(dòng)一個(gè)線程,由線程來(lái)處理客戶端的請(qǐng)求,這樣可以再次監(jiān)聽(tīng) * 下一個(gè)客戶端的連接 */ exec.execute(new ListenrClient(socket)); //通過(guò)線程池來(lái)分配線程 } } catch(Exception e) { e.printStackTrace(); } } /** * 該線程體用來(lái)處理給定的某一個(gè)客戶端的消息,循環(huán)接收客戶端發(fā)送 * 的每一個(gè)字符串,并輸出到控制臺(tái) */ class ListenrClient implements Runnable { private Socket socket; private String name; public ListenrClient(Socket socket) { this.socket = socket; } // 創(chuàng)建內(nèi)部類(lèi)來(lái)獲取昵稱(chēng) private String getName() throws Exception { try { //服務(wù)端的輸入流讀取客戶端發(fā)送來(lái)的昵稱(chēng)輸出流 BufferedReader bReader = new BufferedReader( new InputStreamReader(socket.getInputStream(), "UTF-8")); //服務(wù)端將昵稱(chēng)驗(yàn)證結(jié)果通過(guò)自身的輸出流發(fā)送給客戶端 PrintWriter ipw = new PrintWriter( new OutputStreamWriter(socket.getOutputStream(), "UTF-8"),true); //讀取客戶端發(fā)來(lái)的昵稱(chēng) while(true) { String nameString = bReader.readLine(); if ((nameString.trim().length() == 0) || storeInfo.containsKey(nameString)) { ipw.println("FAIL"); } else { ipw.println("OK"); return nameString; } } } catch(Exception e) { throw e; } } @Override public void run() { try { /* * 通過(guò)客戶端的Socket獲取客戶端的輸出流 * 用來(lái)將消息發(fā)送給客戶端 */ PrintWriter pw = new PrintWriter( new OutputStreamWriter(socket.getOutputStream(), "UTF-8"), true); /* * 將客戶昵稱(chēng)和其所說(shuō)的內(nèi)容存入共享集合HashMap中 */ name = getName(); putIn(name, pw); Thread.sleep(100); // 服務(wù)端通知所有客戶端,某用戶上線 sendToAll("[系統(tǒng)通知] “" + name + "”已上線"); /* * 通過(guò)客戶端的Socket獲取輸入流 * 讀取客戶端發(fā)送來(lái)的信息 */ BufferedReader bReader = new BufferedReader( new InputStreamReader(socket.getInputStream(), "UTF-8")); String msgString = null; while((msgString = bReader.readLine()) != null) { // 檢驗(yàn)是否為私聊(格式:@昵稱(chēng):內(nèi)容) if(msgString.startsWith("@")) { int index = msgString.indexOf(":"); if(index >= 0) { //獲取昵稱(chēng) String theName = msgString.substring(1, index); String info = msgString.substring(index+1, msgString.length()); info = name + ":"+ info; //將私聊信息發(fā)送出去 sendToSomeone(theName, info); continue; } } // 遍歷所有輸出流,將該客戶端發(fā)送的信息轉(zhuǎn)發(fā)給所有客戶端 System.out.println(name+":"+ msgString); sendToAll(name+":"+ msgString); } } catch (Exception e) { // e.printStackTrace(); } finally { remove(name); // 通知所有客戶端,某某客戶已經(jīng)下線 sendToAll("[系統(tǒng)通知] "+name + "已經(jīng)下線了。"); if(socket!=null) { try { socket.close(); } catch(IOException e) { e.printStackTrace(); } } } } } public static void main(String[] args) { TCPServer server = new TCPServer(); server.start(); } }
客戶端:
import java.io.*; import java.net.*; import java.util.Scanner; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.ThreadPoolExecutor; public class TCPClient { static private Socket clientSocket; public TCPClient() {} public static void main(String[] args) throws Exception { Scanner scanner = new Scanner(System.in); String serverIP; System.out.println("請(qǐng)?jiān)O(shè)置服務(wù)器IP:"); serverIP = scanner.next(); clientSocket = new Socket(serverIP, 6789); TCPClient client = new TCPClient(); client.start(); } public void start() { try { Scanner scanner = new Scanner(System.in); setName(scanner); // 接收服務(wù)器端發(fā)送過(guò)來(lái)的信息的線程啟動(dòng) ExecutorService exec = Executors.newCachedThreadPool(); exec.execute(new ListenrServser()); // 建立輸出流,給服務(wù)端發(fā)信息 PrintWriter pw = new PrintWriter( new OutputStreamWriter(clientSocket.getOutputStream(), "UTF-8"), true); while(true) { pw.println(scanner.nextLine()); } } catch(Exception e) { e.printStackTrace(); } finally { if (clientSocket !=null) { try { clientSocket.close(); } catch(IOException e) { e.printStackTrace(); } } } } private void setName(Scanner scan) throws Exception { String name; //創(chuàng)建輸出流 PrintWriter pw = new PrintWriter( new OutputStreamWriter(clientSocket.getOutputStream(), "UTF-8"),true); //創(chuàng)建輸入流 BufferedReader br = new BufferedReader( new InputStreamReader(clientSocket.getInputStream(),"UTF-8")); while(true) { System.out.println("請(qǐng)創(chuàng)建您的昵稱(chēng):"); name = scan.nextLine(); if (name.trim().equals("")) { System.out.println("昵稱(chēng)不得為空"); } else { pw.println(name); String pass = br.readLine(); if (pass != null && (!pass.equals("OK"))) { System.out.println("昵稱(chēng)已經(jīng)被占用,請(qǐng)重新輸入:"); } else { System.out.println("昵稱(chēng)“"+name+"”已設(shè)置成功,可以開(kāi)始聊天了"); break; } } } } // 循環(huán)讀取服務(wù)端發(fā)送過(guò)來(lái)的信息并輸出到客戶端的控制臺(tái) class ListenrServser implements Runnable { @Override public void run() { try { BufferedReader br = new BufferedReader( new InputStreamReader(clientSocket.getInputStream(), "UTF-8")); String msgString; while((msgString = br.readLine())!= null) { System.out.println(msgString); } } catch(Exception e) { e.printStackTrace(); } } } }
運(yùn)行結(jié)果:
開(kāi)始自己的實(shí)現(xiàn)也不是很完整,后來(lái)也是借鑒別人比較好的思想后完善的,權(quán)當(dāng)分享。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java設(shè)計(jì)模式之責(zé)任鏈模式簡(jiǎn)介
這篇文章主要介紹了Java設(shè)計(jì)模式之責(zé)任鏈模式,需要的朋友可以參考下2014-07-07通過(guò)RedisTemplate連接多個(gè)Redis過(guò)程解析
這篇文章主要介紹了通過(guò)RedisTemplate連接多個(gè)Redis過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08MyBatis開(kāi)啟二級(jí)緩存實(shí)現(xiàn)過(guò)程解析
這篇文章主要介紹了MyBatis開(kāi)啟二級(jí)緩存實(shí)現(xiàn)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07dubbo擴(kuò)展點(diǎn)AOP切面功能擴(kuò)展示例詳解
這篇文章主要為大家介紹了dubbo擴(kuò)展點(diǎn)AOP切面功能擴(kuò)展示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08