百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分类 > 正文

SpringBoot+Websocket实现即时通讯

ztj100 2025-01-19 02:00 15 浏览 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}' 来请求连接了

注意:如果后台做了用户认证配置了拦截器需要放行接口

相关推荐

从IDEA开始,迈进GO语言之门(idea got)

前言笔者在学习GO语言编程的时候,GO语言在国内还没有像JAVA/Php/Python那样普及,绕了不少的弯路,要开始入门学习一门编程语言,最好就先从选择一个好的编程语言的开发环境开始,有了这个开发环...

基于SpringBoot+MyBatis的私人影院java网上购票jsp源代码Mysql

本项目为前几天收费帮学妹做的一个项目,JavaEEJSP项目,在工作环境中基本使用不到,但是很多学校把这个当作编程入门的项目来做,故分享出本项目供初学者参考。一、项目介绍基于SpringBoot...

基于springboot的个人服装管理系统java网上商城jsp源代码mysql

本项目为前几天收费帮学妹做的一个项目,JavaEEJSP项目,在工作环境中基本使用不到,但是很多学校把这个当作编程入门的项目来做,故分享出本项目供初学者参考。一、项目介绍基于springboot...

基于springboot的美食网站Java食品销售jsp源代码Mysql

本项目为前几天收费帮学妹做的一个项目,JavaEEJSP项目,在工作环境中基本使用不到,但是很多学校把这个当作编程入门的项目来做,故分享出本项目供初学者参考。一、项目介绍基于springboot...

贸易管理进销存springboot云管货管账分析java jsp源代码mysql

本项目为前几天收费帮学妹做的一个项目,JavaEEJSP项目,在工作环境中基本使用不到,但是很多学校把这个当作编程入门的项目来做,故分享出本项目供初学者参考。一、项目描述贸易管理进销存spring...

SpringBoot+VUE员工信息管理系统Java人员管理jsp源代码Mysql

本项目为前几天收费帮学妹做的一个项目,JavaEEJSP项目,在工作环境中基本使用不到,但是很多学校把这个当作编程入门的项目来做,故分享出本项目供初学者参考。一、项目介绍SpringBoot+V...

目前见过最牛的一个SpringBoot商城项目(附源码)还有人没用过吗

帮粉丝找了一个基于SpringBoot的天猫商城项目,快速部署运行,所用技术:MySQL,Druid,Log4j2,Maven,Echarts,Bootstrap...免费给大家分享出来前台演示...

SpringBoot+Mysql实现的手机商城附带源码演示导入视频

今天为大家带来的是基于SpringBoot+JPA+Thymeleaf框架的手机商城管理系统,商城系统分为前台和后台、前台用的是Bootstrap框架后台用的是SpringBoot+JPA都是现在主...

全网首发!马士兵内部共享—1658页《Java面试突击核心讲》

又是一年一度的“金九银十”秋招大热门,为助力广大程序员朋友“面试造火箭”,小编今天给大家分享的便是这份马士兵内部的面试神技——1658页《Java面试突击核心讲》!...

SpringBoot数据库操作的应用(springboot与数据库交互)

1.JDBC+HikariDataSource...

SpringBoot 整合 Flink 实时同步 MySQL

1、需求在Flink发布SpringBoot打包的jar包能够实时同步MySQL表,做到原表进行新增、修改、删除的时候目标表都能对应同步。...

SpringBoot + Mybatis + Shiro + mysql + redis智能平台源码分享

后端技术栈基于SpringBoot+Mybatis+Shiro+mysql+redis构建的智慧云智能教育平台基于数据驱动视图的理念封装element-ui,即使没有vue的使...

Springboot+Mysql舞蹈课程在线预约系统源码附带视频运行教程

今天发布的是由【猿来入此】的优秀学员独立做的一个基于springboot脚手架的Springboot+Mysql舞蹈课程在线预约系统,系统项目源代码在【猿来入此】获取!https://www.yuan...

SpringBoot+Mysql在线众筹系统源码+讲解视频+开发文档(参考论文

今天发布的是由【猿来入此】的优秀学员独立做的一个基于springboot脚手架的在线众筹管理系统,主要实现了普通用户在线参与众筹基本操作流程的全部功能,系统分普通用户、超级管理员等角色,除基础脚手架外...

Docker一键部署 SpringBoot 应用的方法,贼快贼好用

这两天发现个Gradle插件,支持一键打包、推送Docker镜像。今天我们来讲讲这个插件,希望对大家有所帮助!GradleDockerPlugin简介...

取消回复欢迎 发表评论: