java Socket實(shí)現(xiàn)多人群聊與私聊功能
本文實(shí)例為大家分享了java Socket實(shí)現(xiàn)多人群聊與私聊的具體代碼,供大家參考,具體內(nèi)容如下
關(guān)于Socket套接字的一些基本知識(shí)與認(rèn)識(shí)可以參見上一篇或自行查閱。
ServerSocket和Socket實(shí)現(xiàn)群聊與私聊涉及到多線程編程,實(shí)現(xiàn)過程的重點(diǎn)是利用Socket通信的原理,即不斷的在服務(wù)端和客戶端創(chuàng)建輸入輸出流來相互傳遞、交換數(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)建線程池來管理客戶端的連接線程
* 避免系統(tǒng)資源過度浪費(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è)線程,由線程來處理客戶端的請(qǐng)求,這樣可以再次監(jiān)聽
* 下一個(gè)客戶端的連接
*/
exec.execute(new ListenrClient(socket)); //通過線程池來分配線程
}
} catch(Exception e) {
e.printStackTrace();
}
}
/**
* 該線程體用來處理給定的某一個(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)部類來獲取昵稱
private String getName() throws Exception {
try {
//服務(wù)端的輸入流讀取客戶端發(fā)送來的昵稱輸出流
BufferedReader bReader = new BufferedReader(
new InputStreamReader(socket.getInputStream(), "UTF-8"));
//服務(wù)端將昵稱驗(yàn)證結(jié)果通過自身的輸出流發(fā)送給客戶端
PrintWriter ipw = new PrintWriter(
new OutputStreamWriter(socket.getOutputStream(), "UTF-8"),true);
//讀取客戶端發(fā)來的昵稱
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 {
/*
* 通過客戶端的Socket獲取客戶端的輸出流
* 用來將消息發(fā)送給客戶端
*/
PrintWriter pw = new PrintWriter(
new OutputStreamWriter(socket.getOutputStream(), "UTF-8"), true);
/*
* 將客戶昵稱和其所說的內(nèi)容存入共享集合HashMap中
*/
name = getName();
putIn(name, pw);
Thread.sleep(100);
// 服務(wù)端通知所有客戶端,某用戶上線
sendToAll("[系統(tǒng)通知] “" + name + "”已上線");
/*
* 通過客戶端的Socket獲取輸入流
* 讀取客戶端發(fā)送來的信息
*/
BufferedReader bReader = new BufferedReader(
new InputStreamReader(socket.getInputStream(), "UTF-8"));
String msgString = null;
while((msgString = bReader.readLine()) != null) {
// 檢驗(yàn)是否為私聊(格式:@昵稱:內(nèi)容)
if(msgString.startsWith("@")) {
int index = msgString.indexOf(":");
if(index >= 0) {
//獲取昵稱
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ā)送過來的信息的線程啟動(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)建您的昵稱:");
name = scan.nextLine();
if (name.trim().equals("")) {
System.out.println("昵稱不得為空");
} else {
pw.println(name);
String pass = br.readLine();
if (pass != null && (!pass.equals("OK"))) {
System.out.println("昵稱已經(jīng)被占用,請(qǐng)重新輸入:");
} else {
System.out.println("昵稱“"+name+"”已設(shè)置成功,可以開始聊天了");
break;
}
}
}
}
// 循環(huán)讀取服務(wù)端發(fā)送過來的信息并輸出到客戶端的控制臺(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é)果:

開始自己的實(shí)現(xiàn)也不是很完整,后來也是借鑒別人比較好的思想后完善的,權(quán)當(dāng)分享。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
IDEA修改java文件后 不用重啟Tomcat服務(wù)便可實(shí)現(xiàn)自動(dòng)更新
這篇文章主要介紹了IDEA修改java文件后 不用重啟Tomcat服務(wù)便可實(shí)現(xiàn)自動(dòng)更新,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11
最長(zhǎng)公共子序列問題的深度分析與Java實(shí)現(xiàn)方式
本文詳細(xì)介紹了最長(zhǎng)公共子序列(LCS)問題,包括其概念、暴力解法、動(dòng)態(tài)規(guī)劃解法,并提供了Java代碼實(shí)現(xiàn),暴力解法雖然簡(jiǎn)單,但在大數(shù)據(jù)處理中效率較低,動(dòng)態(tài)規(guī)劃解法通過構(gòu)建DP表,顯著提高了計(jì)算效率,適用于大規(guī)模數(shù)據(jù)處理2025-02-02
SpringCloud?GateWay網(wǎng)關(guān)示例代碼詳解
這篇文章主要介紹了SpringCloud?GateWay網(wǎng)關(guān),Spring?cloud?Gateway的功能很多很強(qiáng)大,文中提到了Spring?Cloud?Gateway中幾個(gè)重要的概念,結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧2022-04-04
SpringBoot Web開發(fā)之系統(tǒng)任務(wù)啟動(dòng)與路徑映射和框架整合
這篇文章主要介紹了SpringBoot Web開發(fā)中的系統(tǒng)任務(wù)啟動(dòng)與路徑映射和Servlet、Filter、Listener框架整合,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
Java?詳解Collection集合之ArrayList和HashSet
本章具體介紹了ArrayList和HashSet兩種集合的基本使用方法和區(qū)別,圖解穿插代碼實(shí)現(xiàn)。?JAVA成仙路從基礎(chǔ)開始講,后續(xù)會(huì)講到JAVA高級(jí),中間會(huì)穿插面試題和項(xiàng)目實(shí)戰(zhàn),希望能給大家?guī)韼椭?/div> 2022-03-03
基于Java編寫一個(gè)數(shù)據(jù)庫(kù)比較工具類
這篇文章主要為大家詳細(xì)介紹了如何基于Java編寫一個(gè)數(shù)據(jù)庫(kù)比較工具類,其中比較結(jié)果會(huì)以現(xiàn)數(shù)據(jù)庫(kù)的視角說明,感興趣的小伙伴可以了解一下2023-07-07
java web開發(fā)之實(shí)現(xiàn)購(gòu)物車功能
這篇文章主要為大家詳細(xì)介紹了java web開發(fā)之實(shí)現(xiàn)購(gòu)物車功能的相關(guān)資料,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-07-07
Java實(shí)現(xiàn)求解一元n次多項(xiàng)式的方法示例
這篇文章主要介紹了Java實(shí)現(xiàn)求解一元n次多項(xiàng)式的方法,涉及java高斯消元法處理矩陣運(yùn)算解多項(xiàng)式的相關(guān)操作技巧,需要的朋友可以參考下2018-01-01最新評(píng)論

