WebSocket实现Web聊天室功能
作者:DOLFAMINGO 发布时间:2023-11-27 06:10:52
标签:WebSocket,Web,聊天室
本文为大家分享了WebSocket实现Web聊天室的具体代码,供大家参考,具体内容如下
一.客户端
JS代码如下:
/*
* 这部分js将websocket封装起来
*/
var websocket = null;
//判断当前浏览器是否支持WebSocket
if ('WebSocket' in window) {
websocket = new WebSocket("ws://localhost:8080/GoodMan/ChatService");
}
else {
alert('当前浏览器 Not support websocket')
}
//连接成功建立的回调方法
websocket.onopen = function () {
alert("WebSocket连接成功");
}
//连接发生错误的回调方法
websocket.onerror = function () {
alert("WebSocket连接发生错误");
};
//发送消息
function sendMess(content) {
var json ="{'username':'"+'${sessionScope.username }'+"', 'content':'"+content+"'}";
websocket.send(json);
}
//接收到消息的回调方法
websocket.onmessage = function (event) {
var jsonString = event.data; //接收到的信息存放在event.data中
var data = JSON.parse(jsonString); //将json字符串转换成js对象
// 然后输出data
}
//连接关闭的回调方法
websocket.onclose = function () {
alert("WebSocket连接关闭");
}
//监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常。
window.onbeforeunload = function () {
closeWebSocket();
}
//关闭WebSocket连接
function closeWebSocket() {
websocket.close();
}
二.服务器
import java.io.IOException;
import java.util.Date;
import java.util.HashSet;
import java.util.Set;
import javax.websocket.*;
import javax.websocket.server.ServerEndpoint;
import com.google.gson.Gson;
class Mess{ //封装一条消息
private String username;
private String content;
private String date;
public Mess(String username, String content, String date) {
super();
this.username = username;
this.content = content;
this.date = date;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getContent() {
return content;
}
public void setContent(String content) {
this.content = content;
}
public String getDate() {
return date;
}
public void setDate(String date) {
this.date = date;
}
}
@ServerEndpoint("/ChatService")
public class ChatService { //每进入一个用户,就新建一个ChatService对象
private static int onlineCount = 0; //静态变量, 用来记录当前在线连接数
private static Set<ChatService> webSocketSet = new HashSet<>(); //静态变量,用来存放在线用户
private Session session; //用于存储与该用户的会话,要通过它来给客户端发送数据
/**
* 连接建立成功调用的方法
* @param session 可选的参数。session为与某个客户端的连接会话,需要通过它来给客户端发送数据
*/
@OnOpen
public void onOpen(Session session){
this.session = session;
webSocketSet.add(this); //加入set中
addOnlineCount(); //在线数加1
System.out.println("有新连接加入!当前在线人数为" + getOnlineCount());
}
/**
* 连接关闭调用的方法
*/
@OnClose
public void onClose(){
webSocketSet.remove(this); //从set中删除
subOnlineCount(); //在线数减1
System.out.println("有一连接关闭!当前在线人数为" + getOnlineCount());
}
/**
* 收到客户端消息后调用的方法
* @param message 客户端发送过来的消息
* @param session 可选的参数
*/
@OnMessage
public void onMessage(String data, Session session) {
Mess mess = new Gson().fromJson(data, Mess.class);
System.out.printf("来%s的消息:%s\n", mess.getUsername(), mess.getContent());
//群发消息
for(ChatService item: webSocketSet){
try {
item.sendMessage(mess);
} catch (IOException e) {
e.printStackTrace();
continue;
}
}
}
/**
* 发生错误时调用
* @param session
* @param error
*/
@OnError
public void onError(Session session, Throwable error){
System.out.println("发生错误");
error.printStackTrace();
}
//以下为辅助函数
public void sendMessage(Mess mess) throws IOException{
String datatime = new java.text.SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(new Date());
mess.setDate(datatime);
String jsonInfo = new Gson().toJson(mess); //将消息对象转换成json字符串
this.session.getAsyncRemote().sendText(jsonInfo); //通过session异步地将信息发送到客户端上
}
public static synchronized int getOnlineCount() {
return onlineCount;
}
public static synchronized void addOnlineCount() {
ChatService.onlineCount++;
}
public static synchronized void subOnlineCount() {
ChatService.onlineCount--;
}
}
来源:https://www.cnblogs.com/DOLFAMINGO/p/9544955.html


猜你喜欢
- 本文实例讲述了Android开发实现标题随scrollview滑动变色的方法。分享给大家供大家参考,具体如下:要实现某个view的背景透明度
- 注解注解为我们在代码中添加信息提供一种形式化的方法,使我们可以在源码、编译时、运行时非常方便的使用这些数据。注解是在JAVA SE5中引入的
- 匿名类是不能有名称的类,所以没办法引用它们。必须在创建时,作为new语句的一部分来声明它们。这就要采用另一种形式的new语句,如下所示: n
- 最近研究OpenCV想用java进行开发,因此研究了一下怎么在Eclipse中配置基于java的Opencv.第一步:先到OpenCV官网下
- 废话不多说,直接上代码import java.text.SimpleDateFormat; import java.util.Date;/*
- 1.场景介绍:开发过程中我们经常性的会用到许多的中间表,用于数据之间的对应和关联.这个时候我们关联最多的就是ID,我们在一张表中插入数据后级
- Spring Boot应用内存飙升一个简单的Spring Boot应用, 几乎只有一个用户在用,内存竟然达到1.2G, 可怕服务现状由于之前
- 记住我功能原理分析还记得前面咱们分析认证流程时,提到的记住我功能吗?现在继续跟踪找到AbstractRememberMeServices对象
- 效果图代码 package com.jh.timelinedemo;import android.content.Context;
- 1、什么是反射?在java开发中有一个非常重要的概念就是java反射机制,也是java的重要特征之一。反射的概念是由Smith在1982年首
- 本文介绍在使用C#开发WinForm程序时,如何使用自定义的XML配置文件。虽然也可以使用app.config,但命名方面很别扭。我们在使用
- 本文实例为大家分享了C#字符串倒序写法的实现代码,供大家参考,具体内容如下//string concatenation with for l
- 一 :问题背景问题:当查询接口较复杂时候,数据的获取都需要[远程调用],必然需要花费更多的时间。 假如查询文章详情页面,需要如下标注的时间才
- gateway网关与前端请求的跨域问题最近因项目需要,引入了gateway网关。可是发现将前端请求的端口指向网关后,用postman发送请求
- 本文实例讲述了Android编程实现为ListView创建上下文菜单(ContextMenu)的方法。分享给大家供大家参考,具体如下:Con
- 本文实例为大家分享了Unity shader实现高斯模糊效果的具体代码,供大家参考,具体内容如下正常图:高斯模糊效果图:shader代码:S
- 为什么不用SQLite? 原因多种:除了面向对象和关系数据库之间的阻抗不匹配时,SQLite可能是矫枉过正(带来了更多的开销)对于一些简单的
- 本文实例讲述了Android编程开发ScrollView中ViewPager无法正常滑动问题解决方法。分享给大家供大家参考,具体如下:这里主
- 递归生成一个如图的菜单,编写两个类数据模型Menu、和创建树形的MenuTree。通过以下过程实现:1.首先从菜单数据中获取所有根节点。2.
- 我们要使用java来操作redis什么是Jedis?什么是Jedis 是Redis官方推荐的java连接开发工具!使用Java操作Redis