java swing+socket实现多人聊天程序
swing+socket实现多人聊天程序
1.准备工作
先看效果:

- 客户端项目结构图:

- 服务端项目结构图:

2.运行原理
- 服务端
先开一个线程serverListerner,线程中开启一个Serversocket
用Serversocket.accept()监听指定端口
一旦有socket连接进来,就为该socket开启一个线程,用于读取该socket输入流的信息,一旦有信息,就通知其他客户端
并将线程保存到Vector<ChartThread> 集合内,交给ChatThreadManager管理,主要是处理分发消息
- 客户端:
点击connect按钮和服务端建立连接,客户端会开启线程随时监听socket的输入流
服务端accept后就会为该socket建立线程,监控该socket的流信息
当客户端点击发送按钮时,将文本框内的信息,写进socket中,此时服务端的线程抓到了socket的输入流信息,就让ChatThreadManager管理类,去通知集合内其他socket,为多个socket写入相同的信息,那么其余的客户端就收到消息了,实现了多人聊天的功能
3.核心代码
- 服务端:
StartServer.java
Main函数入口,实例ServerListener类,并开启该线程
package com.chart;
public class StartServer
{
public static void main(String[] args)
{
new ServerListener().start();
}
}
ServerListener.java:
用Serversocket.accept()监听指定端口,一旦有socket连接进来,就为该socket开启一个ChatThread线程
并将ChatThread线程保存到Vector<ChartThread> 集合内,交给ChatThreadManager管理,
package com.chart;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
import javax.swing.JOptionPane;
public class ServerListener extends Thread{
public void run()
{
try {
ServerSocket serversocket= new ServerSocket(23456);
while(true) {
Socket socket=serversocket.accept(); //进程会阻塞在这句,直到有socket连接进来,就往下执行
System.out.println("one client has connected");
ChatThread cs=new ChatThread(socket);
cs.start();
ChatThreadManager.getChartManager().add(cs);
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
ChatThread:
在此线程中监听客户端socket的来信 并调用publish()方法,通知ChatThreadManager类
package com.chart;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.Socket;
public class ChatThread extends Thread {
Socket socket;
public ChatThread(Socket s) {
this.socket = s;
}
public void out(String out) {
try {
socket.getOutputStream().write(out.getBytes("UTF-8"));
} catch (Exception e) {
}
}
public void run() {
try {
BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream(), "UTF-8"));
String line = "";
while ((line = br.readLine()) != null) {
System.out.println(line);
ChatThreadManager.getChartManager().publish(this, line);
}
br.close();
} catch (Exception e) {
// TODO: handle exception
}
}
}
ChatThreadManager.java
单例类,构造函数私有,管理多个ChatThread, 用于通知其他客户端
package com.chart;
import java.util.Vector;
public class ChatThreadManager {
private ChatThreadManager() {
}
private static final ChatThreadManager Cm = new ChatThreadManager();
public static ChatThreadManager getChartManager() {
return Cm;
}
Vector<ChatThread> ChatSocket_vector = new Vector<ChatThread>();
public void add(ChatThread cs) {
ChatSocket_vector.add(cs);
}
public void publish(ChatThread cs, String msg) {
for (int i = 0; i < ChatSocket_vector.size(); i++) {
ChatThread csTemp = ChatSocket_vector.get(i);
if (!cs.equals(csTemp)) {
csTemp.out(msg + "\n");
}
}
}
}
- 客户端:
StartClient.java:
main 函数入口,新建主窗体,并显示
package com.client;
import java.awt.EventQueue;
import view.MyClientWindow;
public class StartClient {
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
try {
MyClientWindow frame=new MyClientWindow();
frame.setVisible(true);
ConnectionManager.getChatManager().setWindow(frame);
}
catch (Exception e) {
}
}
});
}
}
MyClientWindow.java:
继承JFrame,布局窗体,connect按钮执行建立连接事件,send按钮则往socket中写入发送信息
package view;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import javax.swing.GroupLayout;
import javax.swing.GroupLayout.Alignment;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextArea;
import javax.swing.JTextField;
import javax.swing.LayoutStyle.ComponentPlacement;
import javax.swing.border.EmptyBorder;
import com.client.ConnectionManager;
public class MyClientWindow extends JFrame {
private static final long serialVersionUID = 1L;
private JPanel contentPane;
private JTextArea txt;
private JTextField txtip;
private JTextField txtSend;
public MyClientWindow() {
setAlwaysOnTop(true);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setBounds(100, 100, 450, 300);
contentPane = new JPanel();
contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
setContentPane(contentPane);
txt = new JTextArea();
txt.setText("准备...");
txtip = new JTextField();
txtip.setText("120.254.12.102");
txtip.setColumns(10);
JButton btnConnect = new JButton("connect");
btnConnect.addMouseListener(new MouseAdapter() {
@Override
public void mouseClicked(MouseEvent e) {
ConnectionManager.getChatManager().connect(txtip.getText());
}
});
txtSend = new JTextField();
txtSend.setText("hello");
txtSend.setColumns(10);
JButton btnSend = new JButton("send");
btnSend.addMouseListener(new MouseAdapter() {
@Override
public void mouseClicked(MouseEvent e) {
ConnectionManager.getChatManager().send(txtSend.getText());
appendText("我说: " + txtSend.getText());
txtSend.setText("");
}
});
GroupLayout gl_contentPane = new GroupLayout(contentPane);
gl_contentPane.setHorizontalGroup(gl_contentPane.createParallelGroup(Alignment.LEADING).addGroup(
Alignment.TRAILING,
gl_contentPane.createSequentialGroup().addGroup(gl_contentPane.createParallelGroup(Alignment.TRAILING)
.addGroup(gl_contentPane.createSequentialGroup()
.addComponent(txtSend, GroupLayout.DEFAULT_SIZE, 325, Short.MAX_VALUE)
.addPreferredGap(ComponentPlacement.RELATED)
.addComponent(btnSend, GroupLayout.PREFERRED_SIZE, 109, GroupLayout.PREFERRED_SIZE))
.addGroup(Alignment.LEADING,
gl_contentPane.createSequentialGroup()
.addComponent(txtip, GroupLayout.PREFERRED_SIZE, 294,
GroupLayout.PREFERRED_SIZE)
.addPreferredGap(ComponentPlacement.RELATED)
.addComponent(btnConnect, GroupLayout.DEFAULT_SIZE, 140, Short.MAX_VALUE))
.addComponent(txt, GroupLayout.DEFAULT_SIZE, 434, Short.MAX_VALUE)).addContainerGap()));
gl_contentPane.setVerticalGroup(gl_contentPane.createParallelGroup(Alignment.LEADING)
.addGroup(gl_contentPane.createSequentialGroup()
.addGroup(gl_contentPane.createParallelGroup(Alignment.BASELINE)
.addComponent(txtip, GroupLayout.PREFERRED_SIZE, GroupLayout.DEFAULT_SIZE,
GroupLayout.PREFERRED_SIZE)
.addComponent(btnConnect))
.addPreferredGap(ComponentPlacement.RELATED)
.addComponent(txt, GroupLayout.DEFAULT_SIZE, 198, Short.MAX_VALUE)
.addPreferredGap(ComponentPlacement.RELATED)
.addGroup(gl_contentPane.createParallelGroup(Alignment.TRAILING).addComponent(btnSend)
.addComponent(txtSend, GroupLayout.PREFERRED_SIZE, GroupLayout.DEFAULT_SIZE,
GroupLayout.PREFERRED_SIZE))));
contentPane.setLayout(gl_contentPane);
}
/* 客户端发送的内容添加到中间的txt控件中 */
public void appendText(String in) {
txt.append("\n" + in);
}
}
ConnectionManager.java
package com.client;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.Socket;
import java.net.UnknownHostException;
import view.MyClientWindow;
public class ConnectionManager {
private ConnectionManager() {
}
private static final ConnectionManager instance = new ConnectionManager();
public static ConnectionManager getChatManager() {
return instance;
}
MyClientWindow window;// 为了能在界面上显示服务器发来的信息,就需要传一个MainWindow的引用进来
Socket socket;
private String IP;
BufferedReader bReader;
PrintWriter pWriter;
public void setWindow(MyClientWindow window) {
this.window = window;
}
public void connect(String ip) {
this.IP = ip;
new Thread() {
@Override
public void run() {
// 实现网络方法
try {
socket = new Socket(IP, 23456);
// 输出流
pWriter = new PrintWriter(new OutputStreamWriter(socket.getOutputStream()));
// 输入流
bReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
String line = null;
// 如果读取数据为空
while ((line = bReader.readLine()) != null) {
window.appendText("收到: " + line);
}
// 读完数据之后要关闭
pWriter.close();
bReader.close();
pWriter = null;
bReader = null;
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}.start();
}
public void send(String sendMsg) {
if (pWriter != null) {
pWriter.write(sendMsg + "\n");
pWriter.flush();
} else {
window.appendText("当前链接已经中断...");
}
}
}
4.运行服务端程序:
1.把MainServerPro服务端程序导出成jar包
2.用xshell把jar包上传到linux
3.程序选用了23456端口,因为部署在阿里云的服务器上,所以要打开阿里云的端口限制

4.关闭centos7的防火墙
# service firewalld status; #查看防火墙状态
(disabled 表明 已经禁止开启启动 enable 表示开机自启,inactive 表示防火墙关闭状态 activated(running)表示为开启状态)
# service firewalld start; #开启防火墙
# service firewalld stop; #关闭防火墙
5.启动服务,运行MainServerPro.jar
有以下三种方式,选择第二种:java -jar xxxxx.jar &
java -jar xxxxx.jar // 当前ssh窗口被锁定,可按CTRL + C打断程序运行,或直接关闭窗口,程序退出
java -jar xxxxx.jar & //当前ssh窗口不被锁定,但是当窗口关闭时,程序中止运行。
nohup Java -jar xxxxxx.jar & //不挂断运行命令,当账户退出或终端关闭时,程序仍然运行
6.最后启动客户端,进行聊天
在点击connect的时候出现报错:
Exception in thread "Thread-0" java.awt.HeadlessException: No X11 DISPLAY variable was set, but this program performed an operation which requires it.
原因是因为服务端程序中有一句JOptionPane.showMessageDialog(null, "one client has connect to 12345 port");
在linux这种不支持图形界面的环境中 调用图形界面swing有关的代码时,会抛出异常。。。
解决方法很粗暴,直接把这句代码删掉
5.源代码下载
https://download.csdn.net/download/wcc27857285/10784144
java swing+socket实现多人聊天程序的更多相关文章
- Java使用socket实现两人聊天对话
import java.io.*; import java.net.ServerSocket; import java.net.Socket; import java.util.Scanner; /* ...
- Java网络编程以及简单的聊天程序
网络编程技术是互联网技术中的主流编程技术之一,懂的一些基本的操作是非常必要的.这章主要讲解网络编程,UDP和Socket编程,以及使用Socket做一个简单的聊天软件. 全部代码下载:链接 1.网络编 ...
- java socket之多人聊天室Demo
一.功能介绍 该功能实现了一个类似QQ的最简单多人聊天室,如下图所示. 二.目录结构 三.服务端 1)SocketServer类,该类是服务端的主类,主要负责创建聊天窗口,创建监听客户端的线程: pa ...
- 多线程+socket实现多人聊天室
最近在学习多线程的时候打算做一个简单的多线程socke聊天的程序,结果发现网上的代码都没有完整的实现功能,所以自己实现了一个demo: demo功能大致就是,有一个服务端负责信息转发,多个客户端发送消 ...
- Android IPC机制(五)用Socket实现跨进程聊天程序
1.Socket简介 Socket也称作“套接字“,是在应用层和传输层之间的一个抽象层,它把TCP/IP层复杂的操作抽象为几个简单的接口供应用层调用以实现进程在网络中通信.它分为流式套接字和数据包套接 ...
- 基于Socket实现多人聊天室
当前支持: 1.仅文字 2.加入聊天室提醒 3.退出聊天室提醒 可能出现的BUG: 1.可能出现客户端发送信息后不能及时推送,需要下一个客户端发送信息后一起推送 服务端代码: 1 package co ...
- java-网络通信--socket实现多人聊天(基于命令行)
先编写最简答的服务器 思路 1编写一个实现Runnable接口的静态内部类 ServerC,便于区分每个客户端 1.1 获取客户端数据函数 public String remsg() 1.2 转发消息 ...
- Java 多线程Socket编程通讯--实现聊天室代码
1.创建服务器类 import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; import ja ...
- java版两人聊天程序
server.java import java.io.*; import java.net.*; import java.text.SimpleDateFormat; import java.util ...
随机推荐
- mongodb操作指令(二):索引,聚合,管道
索引 索引本质上是树,最小的值在最左边的叶子上,最大的值在最右边的叶子上,使用索引可以提高查询速度(而不用全表扫描),也可以预防脏数据的插入(如唯一索引) 索引通常能够极大的提高查询的效率, 如果没有 ...
- 小程序开发之wepy框架
ps 本教程使用wepy 1.7+以上的版本 wepy-让小程序支持组件化开发的框架 鹅厂出品,用于开发自家产品的框架还是很良心的,框架设计思路上参照vue,但不是全部照搬,这点要注意. 对微信小程序 ...
- USACO 2009 Open Treasure Cave /// DFS||并查集 oj26215
题目大意: 输入 p,n,t :p为地点数 判断 t 能否回到源点1 接下来n行 每行输入 a b c: a能到达b和c Sample Input 13 6 76 7 82 3 410 11 128 ...
- winform 旋转图片
//img.RotateFlip(RotateFlipType.Rotate90FlipNone); //顺时针旋转90度 RotateFlipType.Rotate90FlipNone //逆时针旋 ...
- minutia cylinder code MCC lSSR 匹配算法
图一 是LSS匹配算法, 图二是LSSR 匹配算法,数据采用MCC SDK自带的十个人的数据.LSS EER6.0%左右,LSSR EER 0%
- Python全栈开发:模块
模块,用一砣代码实现了某个功能的代码集合. 参考资源:http://www.cnblogs.com/alex3714/articles/5161349.html 类似于函数式编程和面向过程编程,函数式 ...
- 「题解」:[BZOJ4358]permu
问题: permu 时间限制: 30 Sec 内存限制: 512 MB 题面 题目描述 给出一个长度为n的排列P(P1,P2,...Pn),以及m个询问.每次询问某个区间[l,r]中,最长的值域 连 ...
- 一个四五年的Java开发程序员,该准备哪些去面试?
上周面试了一周,感触颇深,总结一下. 面试了公司大概有阿里,携程,爱奇艺,唯品会,途牛,bilibili,大众点评,阿里和爱奇艺是电话面试,其他现场面试. 首先,五年左右,应该算高级开发工程师,大部分 ...
- mybtais分批insert
这里自己写了个对集合按一批的数量进行分批操作的分页bean,见PagenationUtil如下: package com.util; import java.util.List ; /** * @au ...
- java基础之DateFormat类
DateFormat DateFormat类概述 DateFormat 是日期/时间格式化子类的抽象类,它以与语言无关的方式格式化并解析日期或时间. 是抽象类,所以使用其子类SimpleDateFor ...