SpringBoot+Websocket实现即时通讯
ztj100 2025-01-19 02:00 60 浏览 0 评论
1. 介绍
基于Spring Boot和websocket实现点对点在线聊天和简单的机器人自动回复功能,学生可选择在线空闲的老师咨询,无在线老师接入机器人根据学生提问的内容和问题编号从问题库中获取对应的答案来回复学生。
2.后台
2.1. 引入依赖
// https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-websocket
implementation group: 'org.springframework.boot', name: 'spring-boot-starter-websocket', version: '2.0.4.RELEASE'
2.2. websocket 配置类
WebsocketConfig.java
@Configuration
public class WebSocketConfig {
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
}
2.3. 封装显示消息对象
@Data
public class MessageVo {
@ApiModelProperty(value = "用户id")
private Integer userId;
@ApiModelProperty(value = "用户名")
private String username;
@ApiModelProperty(value = "消息内容")
private String msg;
@ApiModelProperty(value = "在线人数")
private int count;
}
2.4. 封装进行对话的对象
用来存放会话的用户信息
@Data
@ApiModel(description = "websocket会话对象")
public class SocketUserInfo extends BaseEntity {
//用户sessionId
private String sessionId;
//用户session
private Session session;
//目标用户sessionid
private String targetSessionId;
//用户角色
private String userRole;
//用户id
private Integer userId;
//用户名
private String userName;
//联系电话
private String tel;
2.5. websocket处理类
@Slf4j
@Component
@ServerEndpoint(value = "/groupChat/{targetSessionId}/{userId}",configurator = SpringContextUtil.class)
public class WebSocketServerController{
//用本地线程保存session
private static ThreadLocal<Session> sessions = new ThreadLocal<Session>();
//保存所有连接上的用户的session sessionId作为key
private static Map<String, SocketUserInfo> userSessionMap = new ConcurrentHashMap<>();
//保存老师的session
private static Map<String, SocketUserInfo> serverSessionMap = new ConcurrentHashMap<>();
//保存在线的学生
private static List<Map<String,Object>> studentUserMapList = new ArrayList<>();
//保存在线空闲的老师信息
private static List<Map<String,Object>> teacherUserMapList = new ArrayList<>();
@Autowired
private CacheProxy cacheProxy;
@Autowired
private BuiProblemAnswerServiceImpl buiProblemAnswerService;
/**
* 建立连接调用的方法,群成员加入
*
* @param session
* @param targetSessionId 学生咨询的对象的session id
*/
@OnOpen
public void onOpen(Session session,
@PathParam("targetSessionId") String targetSessionId,
@PathParam("userId") Integer userId) {
try {
MessageVo messageVo = new MessageVo();
//保证各个线程里的变量相对独立于其他线程内的变量
sessions.set(session);
//获取用户信息
SysLoginUserServiceImpl sysLoginUserService = SpringContextUtil.getBean(SysLoginUserServiceImpl.class);
Map<String, Object> map = sysLoginUserService.queryUserInfoById(userId);
//老师上线
if ("10000".equals(targetSessionId)){
//如果是老师和管理员角色,创建一个在线聊天信息
SocketUserInfo serverInfo = new SocketUserInfo();
serverInfo.setUserRole("老师");
serverInfo.setSession(session);
serverInfo.setUserId(userId);
serverInfo.setSessionId(session.getId());
serverInfo.setUserName(StrUtil.clearNull(map.get("userName")));
serverInfo.setTerminalCode(StrUtil.clearNull(map.get("tel")));
map.put("sessionId",session.getId());
//将老师信息存入到list中
teacherUserMapList.add(map);
//将在线老师信息写入缓存
cacheProxy.set("websocket",teacherUserMapList);
messageVo.setMsg(session.getId()+"您已经上线了");
messageVo.setUserId(userId);
sendMsg(session,messageVo);
//将老师的信息保存到map中
serverSessionMap.put(session.getId(),serverInfo);
}
if (!"10000".equals(targetSessionId)){//学生上线 创建一个在线学生信息
SocketUserInfo userInfo = new SocketUserInfo();
userInfo.setSessionId(session.getId());
userInfo.setSession(session);
userInfo.setUserRole("学生");
userInfo.setUserId(userId);
userInfo.setUserName(StrUtil.clearNull(map.get("userName")));
userInfo.setTerminalCode(StrUtil.clearNull(map.get("tel")));
//获取对应的老师信息
if (!targetSessionId.equals("kefu")){
SocketUserInfo serverInfo = serverSessionMap.get(targetSessionId);
//将该老师的连接设置为当前学生
serverInfo.setTargetSessionId(session.getId());
//将当前学生的连接对象设置为该老师
userInfo.setTargetSessionId(targetSessionId);
//将当前咨询的老师从空闲的列表中删除
for (int i = 0; i < teacherUserMapList.size(); i++) {
Map<String, Object> teacherMap = teacherUserMapList.get(i);
if (teacherMap.get("sessionId").equals(targetSessionId)){
teacherUserMapList.remove(i);
}
}
//更新可咨询的老师列表
cacheProxy.set("websocket",teacherUserMapList);
//给学生发送一条信息
messageVo.setMsg("老师"+serverInfo.getUserName()+"正在为你答疑!");
messageVo.setCount(2);
messageVo.setUserId(serverInfo.getUserId());
sendMsg(userInfo.getSession(), messageVo);
//给老师发送一条信息
messageVo.setMsg("正在为学生"+userInfo.getSessionId()+"答疑!");
messageVo.setUserId(userInfo.getUserId());
sendMsg(serverInfo.getSession(), messageVo);
}
if ("kefu".equals(targetSessionId)){
userInfo.setTargetSessionId("kefu");
messageVo.setMsg("请根据右侧的列表输入要您咨询的问题编号!!!");
sendMsg(session,messageVo);
}
//将在线用户信息保存到map中
userSessionMap.put(session.getId(), userInfo);
}
}catch (Exception e){
e.printStackTrace();
}
}
/**
* 收到消息调用的方法,群成员发送消息
*
* @param message
* @param session
*/
@OnMessage
public void onMessage(String message, Session session){
HashMap<String, String> result = new HashMap<>();
MessageVo messageVo = new MessageVo();
SocketUserInfo serverInfo = serverSessionMap.get(session.getId());
if (serverInfo != null) {//老师消息
log.info("老师"+ session.getId()+"发送消息:\""+ message +"\"给学生"+serverSessionMap.get(session.getId()).getTargetSessionId());
result.put("msg", "老师"+session.getId()+":"+message);
messageVo.setMsg(message);
messageVo.setUserId(serverInfo.getUserId());
messageVo.setCount(2);
//要判断是否绑定到的学生如果有就将消息传递到学生
if (null != serverInfo.getTargetSessionId()){
//将消息发送给绑定的学生
sendMsg(userSessionMap.get(serverSessionMap.get(session.getId()).getTargetSessionId()).getSession(), messageVo);
}
sendMsg(session,messageVo);
} else {//学生消息
System.out.println("学生"+ session.getId()+"发送消息:\""+ message +"\"给老师"+userSessionMap.get(session.getId()).getTargetSessionId());
result.put("msg", "学生"+session.getId()+":"+message);
messageVo.setCount(2);
messageVo.setMsg(message);
messageVo.setUserId(userSessionMap.get(session.getId()).getUserId());
//学生发出一条消息
sendMsg(session,messageVo);
//判断是否绑定了老师,如果有就发送消息
String targetSessionId = userSessionMap.get(session.getId()).getTargetSessionId();
if (null != targetSessionId && !targetSessionId.equals("kefu")){
//给当前学生咨询的老师发送消息内容
messageVo.setUserId(userSessionMap.get(session.getId()).getUserId());
sendMsg(serverSessionMap.get(userSessionMap.get(session.getId()).getTargetSessionId()).getSession(), messageVo);
}
if (null != targetSessionId && targetSessionId.equals("kefu")){
try {
Long problemNumber = Long.valueOf(StrUtil.clearNull(message));
String answer = buiProblemAnswerService.getAnswer(problemNumber);
messageVo.setMsg(answer);
if (answer == null){
messageVo.setMsg("请输入正确的问题编号!!!");
}
}catch (Exception e){
List<BuiProblemAnswer> buiProblemAnswerList = buiProblemAnswerService.getAnswerList(message);
if (buiProblemAnswerList.size() > 1){
String msg = "猜你想问:\t";
for (BuiProblemAnswer buiProblemAnswer : buiProblemAnswerList) {
msg = msg + buiProblemAnswer.getProblemTopic()+"\t";
}
messageVo.setMsg(msg);
}else if (buiProblemAnswerList.size() == 1){
messageVo.setMsg(buiProblemAnswerList.get(0).getProblemAnswer());
}else if (buiProblemAnswerList.size() == 0){
messageVo.setMsg("抱歉!没有找到您的问题!");
}
// messageVo.setMsg("猜你想问:\t"+);
// messageVo.setMsg("请输入正确的问题编号!!!");
}
}
//回复学生消息
messageVo.setUserId(null);
sendMsg(session,messageVo);
}
}
/**
* 关闭连接调用的方法,群成员退出
*
* @param session
* @param targetSessionId
*/
@OnClose
public void onClose(Session session, @PathParam("targetSessionId") String targetSessionId, @PathParam("userId") Integer userId) {
SocketUserInfo serverInfo = serverSessionMap.get(session.getId());
MessageVo messageVo = new MessageVo();
if (serverInfo != null){
//老师下线,将老师从map中移除
for (int i = 0; i < teacherUserMapList.size(); i++) {
Map<String, Object> map = teacherUserMapList.get(i);
if (userId == map.get("userId")){
teacherUserMapList.remove(i);
}
}
//更新缓存中的在线老师的数据
cacheProxy.set("websocket",teacherUserMapList);
serverSessionMap.remove(session.getId());
//查看当前是否有在服务的对象,如果有就给用户发送系统错误的信息
if (serverInfo.getTargetSessionId() != null){
messageVo.setMsg("系统错误,老师已经下线了");
messageVo.setUserId(serverInfo.getUserId());
sendMsg(userSessionMap.get(serverInfo.getTargetSessionId()).getSession(),messageVo);
}
log.info("老师编号:" + session.getId() + "退出了连接,当前在线老师共计:" + serverSessionMap.size());
}else if ("10000".equals(targetSessionId)){
//学生下线 从老师中解绑,解绑后,如果还有在排队的学生,则让这个学生和当前老师绑定起来
userSessionMap.remove(session.getId());
//获取当前老师信息
SocketUserInfo serverUserInfo = serverSessionMap.get(targetSessionId);
HashMap<String, Object> teacherMap = new HashMap<>();
teacherMap.put("userName",serverUserInfo.getUserName());
teacherMap.put("serssionId",serverUserInfo.getSessionId());
teacherMap.put("tel",serverUserInfo.getTerminalCode());
teacherMap.put("userId",serverUserInfo.getUserId());
//将当前老师添加到空闲老师列表并更新缓存
teacherUserMapList.add(teacherMap);
cacheProxy.set("websocket",teacherUserMapList);
log.info("用户编号:" + session.getId() + "退出了连接,当前在线学生共计:" + userSessionMap.size());
}
}
/**
* 传输消息错误调用的方法
*
* @param error
*/
@OnError
public void OnError(Throwable error) {
log.info("Connection error");
}
//查询排队用户
private synchronized String findLineUser(){
//判断是否有用户
if (userSessionMap.size() > 0){
//遍历所有用户,查找一个排队的用户
for (SocketUserInfo UserInfo: userSessionMap.values()) {
if (null == UserInfo.getTargetSessionId()){
return UserInfo.getSessionId();
}
}
}
return null;
}
//查询在线空闲的老师
private synchronized String findFreeServer(){
//判断是否有老师
if (serverSessionMap.size() > 0){
//遍历在线的所有老师,查找一个空闲的老师
for (SocketUserInfo serverInfo: serverSessionMap.values()) {
if (null == serverInfo.getTargetSessionId()){
return serverInfo.getSessionId();
}
}
}
return null;
}
//统一的发送消息方法
private synchronized void sendMsg(Session session, MessageVo messageVo) {
try {
session.getBasicRemote().sendText(JSON.toJSONString(messageVo));
} catch (IOException e) {
e.printStackTrace();
}
}
}
websocket中无法注入bean实例的问题: 解决:
这样就可以使用@Autowired来注入实例了
最后前端就可以使用 'ws://ip地址:端口/项目地址/groupChat/{参数1}/{参数2}' 来请求连接了
注意:如果后台做了用户认证配置了拦截器需要放行接口
相关推荐
- SpringBoot整合SpringSecurity+JWT
-
作者|Sans_https://juejin.im/post/5da82f066fb9a04e2a73daec一.说明SpringSecurity是一个用于Java企业级应用程序的安全框架,主要包含...
- 「计算机毕设」一个精美的JAVA博客系统源码分享
-
前言大家好,我是程序员it分享师,今天给大家带来一个精美的博客系统源码!可以自己买一个便宜的云服务器,当自己的博客网站,记录一下自己学习的心得。开发技术博客系统源码基于SpringBoot,shiro...
- springboot教务管理系统+微信小程序云开发附带源码
-
今天给大家分享的程序是基于springboot的管理,前端是小程序,系统非常的nice,不管是学习还是毕设都非常的靠谱。本系统主要分为pc端后台管理和微信小程序端,pc端有三个角色:管理员、学生、教师...
- SpringBoot+LayUI后台管理系统开发脚手架
-
源码获取方式:关注,转发之后私信回复【源码】即可免费获取到!项目简介本项目本着避免重复造轮子的原则,建立一套快速开发JavaWEB项目(springboot-mini),能满足大部分后台管理系统基础开...
- Spring Boot的Security安全控制——认识SpringSecurity!
-
SpringBoot的Security安全控制在Web项目开发中,安全控制是非常重要的,不同的人配置不同的权限,这样的系统才安全。最常见的权限框架有Shiro和SpringSecurity。Shi...
- 前同事2024年接私活已入百万,都是用这几个开源的SpringBoot项目
-
前言不得不佩服SpringBoot的生态如此强大,今天给大家推荐几款优秀的后台管理系统,小伙伴们再也不用从头到尾撸一个项目了。SmartAdmin...
- 值得学习的15 个优秀开源的 Spring Boot 学习项目
-
SpringBoot算是目前Java领域最火的技术栈了,除了书呢?当然就是开源项目了,今天整理15个开源领域非常不错的SpringBoot项目供大家学习,参考。高富帅的路上只能帮你到这里了,...
- 开发企业官网就用这个基于SpringBoot的CMS系统,真香
-
前言推荐这个项目是因为使用手册部署手册非常...
- 2021年超详细的java学习路线总结—纯干货分享
-
本文整理了java开发的学习路线和相关的学习资源,非常适合零基础入门java的同学,希望大家在学习的时候,能够节省时间。纯干货,良心推荐!第一阶段:Java基础...
- jeecg-boot学习总结及使用心得(jeecgboot简单吗)
-
jeecg-boot学习总结及使用心得1.jeecg-boot是一个真正前后端分离的模版项目,便于二次开发,使用的都是较流行的新技术,后端技术主要有spring-boot2.x、shiro、Myb...
- 后勤集团原料管理系统springboot+Layui+MybatisPlus+Shiro源代码
-
本项目为前几天收费帮学妹做的一个项目,JavaEEJSP项目,在工作环境中基本使用不到,但是很多学校把这个当作编程入门的项目来做,故分享出本项目供初学者参考。一、项目描述后勤集团原料管理系统spr...
- 白卷开源SpringBoot+Vue的前后端分离入门项目
-
简介白卷是一个简单的前后端分离项目,主要采用Vue.js+SpringBoot技术栈开发。除了用作入门练习,作者还希望该项目可以作为一些常见Web项目的脚手架,帮助大家简化搭建网站的流程。...
- Spring Security 自动踢掉前一个登录用户,一个配置搞定
-
登录成功后,自动踢掉前一个登录用户,松哥第一次见到这个功能,就是在扣扣里边见到的,当时觉得挺好玩的。自己做开发后,也遇到过一模一样的需求,正好最近的SpringSecurity系列正在连载,就结...
- 收藏起来!这款开源在线考试系统,我爱了
-
大家好,我是为广大程序员兄弟操碎了心的小编,每天推荐一个小工具/源码,装满你的收藏夹,每天分享一个小技巧,让你轻松节省开发效率,实现不加班不熬夜不掉头发,是我的目标!今天小编推荐一款基于Spr...
- Shiro框架:认证和授权原理(shiro权限认证流程)
-
优质文章,及时送达前言Shiro作为解决权限问题的常用框架,常用于解决认证、授权、加密、会话管理等场景。本文将对Shiro的认证和授权原理进行介绍:Shiro可以做什么?、Shiro是由什么组成的?举...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- SpringBoot整合SpringSecurity+JWT
- 「计算机毕设」一个精美的JAVA博客系统源码分享
- springboot教务管理系统+微信小程序云开发附带源码
- SpringBoot+LayUI后台管理系统开发脚手架
- Spring Boot的Security安全控制——认识SpringSecurity!
- 前同事2024年接私活已入百万,都是用这几个开源的SpringBoot项目
- 值得学习的15 个优秀开源的 Spring Boot 学习项目
- 开发企业官网就用这个基于SpringBoot的CMS系统,真香
- 2021年超详细的java学习路线总结—纯干货分享
- jeecg-boot学习总结及使用心得(jeecgboot简单吗)
- 标签列表
-
- idea eval reset (50)
- vue dispatch (70)
- update canceled (42)
- order by asc (53)
- spring gateway (67)
- 简单代码编程 贪吃蛇 (40)
- transforms.resize (33)
- redisson trylock (35)
- 卸载node (35)
- np.reshape (33)
- torch.arange (34)
- npm 源 (35)
- vue3 deep (35)
- win10 ssh (35)
- vue foreach (34)
- idea设置编码为utf8 (35)
- vue 数组添加元素 (34)
- std find (34)
- tablefield注解用途 (35)
- python str转json (34)
- java websocket客户端 (34)
- tensor.view (34)
- java jackson (34)
- vmware17pro最新密钥 (34)
- mysql单表最大数据量 (35)