欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java Socket實(shí)現(xiàn)多人群聊與私聊功能

 更新時間:2022年05月19日 14:13:18   作者:OneDay-X  
這篇文章主要為大家詳細(xì)介紹了java Socket實(shí)現(xiàn)多人群聊與私聊功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下

本文實(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("請設(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)文章

  • Java設(shè)計模式之責(zé)任鏈模式簡介

    Java設(shè)計模式之責(zé)任鏈模式簡介

    這篇文章主要介紹了Java設(shè)計模式之責(zé)任鏈模式,需要的朋友可以參考下
    2014-07-07
  • 詳解Java如何創(chuàng)建Annotation

    詳解Java如何創(chuàng)建Annotation

    在本文中,我們將介紹注解的基礎(chǔ)知識,包括注解是什么,它們?nèi)绾卧谑纠惺褂茫约叭绾翁幚硭鼈?。下面和小編一起來學(xué)習(xí)吧
    2019-05-05
  • 通過RedisTemplate連接多個Redis過程解析

    通過RedisTemplate連接多個Redis過程解析

    這篇文章主要介紹了通過RedisTemplate連接多個Redis過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-08-08
  • java簡單操作word實(shí)例

    java簡單操作word實(shí)例

    這篇文章主要為大家詳細(xì)介紹了java簡單操作word實(shí)例,感興趣的朋友可以參考一下
    2016-03-03
  • java中final修飾符的使用方法

    java中final修飾符的使用方法

    這篇文章主要為大家詳細(xì)介紹了java中final修飾符的使用方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-07-07
  • SpringCloud Eureka搭建的方法步驟

    SpringCloud Eureka搭建的方法步驟

    這篇文章主要介紹了SpringCloud Eureka搭建的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-01-01
  • Java中不可或缺的關(guān)鍵字volatile詳析

    Java中不可或缺的關(guān)鍵字volatile詳析

    volatile是Java提供的一種輕量級的同步機(jī)制,下面這篇文章主要給大家介紹了關(guān)于Java中不可或缺的關(guān)鍵字volatile的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-12-12
  • MyBatis開啟二級緩存實(shí)現(xiàn)過程解析

    MyBatis開啟二級緩存實(shí)現(xiàn)過程解析

    這篇文章主要介紹了MyBatis開啟二級緩存實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-07-07
  • SpringCloud中Eureka的配置及使用講解

    SpringCloud中Eureka的配置及使用講解

    Eureka?服務(wù)注冊中心,主要用于提供服務(wù)注冊功能,當(dāng)微服務(wù)啟動時,會將自己的服務(wù)注冊到?Eureka?Server,這篇文章主要介紹了SpringCloud中Eureka的配置及詳細(xì)使用,需要的朋友可以參考下
    2023-01-01
  • dubbo擴(kuò)展點(diǎn)AOP切面功能擴(kuò)展示例詳解

    dubbo擴(kuò)展點(diǎn)AOP切面功能擴(kuò)展示例詳解

    這篇文章主要為大家介紹了dubbo擴(kuò)展點(diǎn)AOP切面功能擴(kuò)展示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08

最新評論