java實(shí)現(xiàn)小型局域網(wǎng)群聊功能(C/S模式)
本文實(shí)例為大家分享了java群聊功能,供大家參考,具體內(nèi)容如下
Java 對TCP協(xié)議的支持:
--> java.net包中定義了兩個(gè)類ServerSocket 和Socket ,分別用來實(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è)線程可能會對集合進(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 類對連接到服務(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
* 存儲已連接客戶端的集合
*/
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ù)器端的控制臺
// 把這個(gè)客戶端說的話,發(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ù)端...
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
解決idea打包成功但是resource下的文件沒有成功的問題
這篇文章主要介紹了解決idea打包成功但是resource下的文件沒有成功的問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-08-08
Kotlin基本類型自動(dòng)裝箱出現(xiàn)問題解決辦法
這篇文章主要介紹了Kotlin基本類型自動(dòng)裝箱出現(xiàn)問題解決辦法的相關(guān)資料,希望通過本文能幫助到大家,讓大家遇到這樣的問題順利解決,需要的朋友可以參考下2017-10-10
java通過Idea遠(yuǎn)程一鍵部署springboot到Docker詳解
這篇文章主要介紹了java通過Idea遠(yuǎn)程一鍵部署springboot到Docker詳解,Idea是Java開發(fā)利器,springboot是Java生態(tài)中最流行的微服務(wù)框架,docker是時(shí)下最火的容器技術(shù),那么它們結(jié)合在一起會產(chǎn)生什么化學(xué)反應(yīng)呢?的相關(guān)資料2019-06-06
重寫equals的同時(shí)為何要重寫hashCode?
這篇文章主要給大家介紹了關(guān)于重寫equals的同時(shí)為何要重寫hashCode的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01
Elasticsearch 基礎(chǔ)介紹及索引原理分析
這篇文章主要介紹了Elasticsearch 基礎(chǔ)介紹及索引原理分析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-07-07

