java Socket實(shí)現(xiàn)多人群聊與私聊功能
本文實(shí)例為大家分享了java Socket實(shí)現(xiàn)多人群聊與私聊的具體代碼,供大家參考,具體內(nèi)容如下
關(guān)于Socket套接字的一些基本知識與認(rèn)識可以參見上一篇或自行查閱。
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); //將對應(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() + "”連接成功! ");
/**
* 啟動一個線程,由線程來處理客戶端的請求,這樣可以再次監(jiān)聽
* 下一個客戶端的連接
*/
exec.execute(new ListenrClient(socket)); //通過線程池來分配線程
}
} catch(Exception e) {
e.printStackTrace();
}
}
/**
* 該線程體用來處理給定的某一個客戶端的消息,循環(huán)接收客戶端發(fā)送
* 的每一個字符串,并輸出到控制臺
*/
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("請?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ā)送過來的信息的線程啟動
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("請創(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)被占用,請重新輸入:");
} else {
System.out.println("昵稱“"+name+"”已設(shè)置成功,可以開始聊天了");
break;
}
}
}
}
// 循環(huán)讀取服務(wù)端發(fā)送過來的信息并輸出到客戶端的控制臺
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)分享。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
MyBatis開啟二級緩存實(shí)現(xiàn)過程解析
這篇文章主要介紹了MyBatis開啟二級緩存實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-07-07
dubbo擴(kuò)展點(diǎn)AOP切面功能擴(kuò)展示例詳解
這篇文章主要為大家介紹了dubbo擴(kuò)展點(diǎn)AOP切面功能擴(kuò)展示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08

