java實(shí)現(xiàn)小型局域網(wǎng)群聊功能(C/S模式)
本文實(shí)例為大家分享了java群聊功能,供大家參考,具體內(nèi)容如下
Java 對(duì)TCP協(xié)議的支持:
--> java.net包中定義了兩個(gè)類ServerSocket 和Socket ,分別用來(lái)實(shí)現(xiàn)雙向連接的server 端和client 端。
--> Client 類定義客戶端
package com.dragon.java.tcpchat; import java.io.IOException; import java.net.Socket; import java.net.UnknownHostException; /** * 客戶端 * * @author Auser * */ public class Client { public static void main(String args[]) throws UnknownHostException, IOException { Socket client = new Socket("192.168.1.188", 10000); // 穿件發(fā)送信息的線程 new ClientSend(client).start(); // 創(chuàng)建接收信息的線程 new ClientReceive(client).start(); // 因?yàn)橐獙?shí)現(xiàn)聊天功能,而不是只發(fā)送一次信息,所以輸出流和客戶端都不能關(guān)閉。 // client.shutdownOutput(); // client.close(); } }
--> ClientSend 類 定義客戶端向服務(wù)端發(fā)送信息的線程
package com.dragon.java.tcpchat; import java.io.IOException; import java.io.PrintStream; import java.net.Socket; import java.util.Scanner; /** * 客戶端向服務(wù)端發(fā)送信息的線程 * * @author Auser * */ public class ClientSend extends Thread { private Scanner scanner; private Socket socket; public ClientSend(Socket socket) { this.socket = socket; } @Override public void run() { scanner = new Scanner(System.in); try { PrintStream ps = new PrintStream(socket.getOutputStream()); String line = ""; // 阻塞式發(fā)送信息 while ((line = scanner.nextLine()) != null) { ps.println(line); } } catch (IOException e) { e.printStackTrace(); } } }
--> ClientReceive 類 定義客戶端接收服務(wù)端信息的線程
package com.dragon.java.tcpchat; import java.io.BufferedReader; import java.io.InputStreamReader; import java.net.Socket; /** * 客戶端接收信息的線程 * * @author Auser * */ public class ClientReceive extends Thread { private Socket socket; public ClientReceive(Socket socket) { this.socket = socket; } @Override public void run() { try { BufferedReader br = new BufferedReader(new InputStreamReader( socket.getInputStream())); // 按行接收信息 String line = ""; while ((line = br.readLine()) != null) { System.out.println(line); } } catch (Exception e) { e.printStackTrace(); } } }
--> Server 類定義服務(wù)端
package com.dragon.java.tcpchat; import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; import java.util.ArrayList; import java.util.List; /** * 服務(wù)器 * * @author Auser * */ public class Server { public static void main(String[] args) throws IOException, InterruptedException { List<Socket> list = new ArrayList<>(); // 創(chuàng)建服務(wù)器端的套接字 ServerSocket server = new ServerSocket(10000); while (true) { // 接收客戶端的阻塞方法 Socket socket = server.accept(); // 設(shè)計(jì)到多個(gè)線程可能會(huì)對(duì)集合進(jìn)行增刪的操作,要進(jìn)行同步的處理 synchronized (list) { list.add(socket); } // 啟動(dòng)一個(gè)新的線程去處理這個(gè)客戶端的交流 new HandleSocket(socket, list).start(); } // 因?yàn)椴恢揽蛻舳耸裁磿r(shí)候發(fā)送信息,所以服務(wù)端要一直開啟不能關(guān)閉。 } }
--> HandleSocket 類對(duì)連接到服務(wù)端的客戶端進(jìn)行操作(上下線通知、屏蔽拉黑、將信息發(fā)送到每個(gè)客戶端等...)
package com.dragon.java.tcpchat; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.io.PrintStream; import java.net.InetAddress; import java.net.Socket; import java.util.List; /** * 處理每個(gè)(單個(gè))連接到服務(wù)器的客戶端的線程 * * @author Auser * */ public class HandleSocket extends Thread { private Socket socket; private List<Socket> list; /** * 構(gòu)造方法 * * @param socket * 當(dāng)前連接的客戶端 * @param list * 存儲(chǔ)已連接客戶端的集合 */ public HandleSocket(Socket socket, List<Socket> list) { this.socket = socket; this.list = list; } /** * 線程run方法 */ @Override public void run() { InetAddress address = socket.getInetAddress(); // 獲取連接到服務(wù)器的這的客戶端的地址 String ip = address.getHostAddress(); System.out.println(ip + "上線了!"); if (ip.equals("192.168.1.117")) { synchronized (list) { sendToAll(ip + "由于違規(guī)操作,已被拉入黑名單!"); list.remove(socket); } return; } try { BufferedReader br = new BufferedReader(new InputStreamReader( socket.getInputStream(), "gbk")); String line = ""; while ((line = br.readLine()) != null) { String msg = ip + ":" + line; System.out.println(msg); // 輸出到服務(wù)器端的控制臺(tái) // 把這個(gè)客戶端說(shuō)的話,發(fā)給其他所有的客戶端 sendToAll(msg); } } catch (IOException e) { // e.printStackTrace(); System.out.println(ip + "下線了!"); synchronized (list) { list.remove(socket); } } } /** * 把信息發(fā)送給所有的客戶端,去除當(dāng)前socket * * @param msg * 發(fā)送的信息 */ private void sendToAll(String msg) { synchronized (list) { for (Socket s : list) { if (s != socket) { try { PrintStream ps = new PrintStream(s.getOutputStream()); ps.println(); } catch (IOException e) { e.printStackTrace(); } } } } } }
--> 注意:因?yàn)橐獙?shí)現(xiàn)客戶端連接到服務(wù)端,即客戶端要先找到服務(wù)端,所以服務(wù)端要先開啟,再開啟服務(wù)端...
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
解決idea打包成功但是resource下的文件沒有成功的問(wèn)題
這篇文章主要介紹了解決idea打包成功但是resource下的文件沒有成功的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-08-08Kotlin基本類型自動(dòng)裝箱出現(xiàn)問(wèn)題解決辦法
這篇文章主要介紹了Kotlin基本類型自動(dòng)裝箱出現(xiàn)問(wèn)題解決辦法的相關(guān)資料,希望通過(guò)本文能幫助到大家,讓大家遇到這樣的問(wèn)題順利解決,需要的朋友可以參考下2017-10-10java通過(guò)Idea遠(yuǎn)程一鍵部署springboot到Docker詳解
這篇文章主要介紹了java通過(guò)Idea遠(yuǎn)程一鍵部署springboot到Docker詳解,Idea是Java開發(fā)利器,springboot是Java生態(tài)中最流行的微服務(wù)框架,docker是時(shí)下最火的容器技術(shù),那么它們結(jié)合在一起會(huì)產(chǎn)生什么化學(xué)反應(yīng)呢?的相關(guān)資料2019-06-06重寫equals的同時(shí)為何要重寫hashCode?
這篇文章主要給大家介紹了關(guān)于重寫equals的同時(shí)為何要重寫hashCode的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01Elasticsearch 基礎(chǔ)介紹及索引原理分析
這篇文章主要介紹了Elasticsearch 基礎(chǔ)介紹及索引原理分析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-07-07