IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 网络协议 -> SpringBoot整合WebSocket,实现后台向前端推送信息 -> 正文阅读

[网络协议]SpringBoot整合WebSocket,实现后台向前端推送信息

maven依赖:

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-websocket</artifactId>
        </dependency>

WebSocketConfig:

@Component
public class WebSocketConfig {
    @Bean
    public ServerEndpointExporter serverEndpointExporter(){
        return new ServerEndpointExporter();
    }
}

WebSocketServer:

  1. 因为WebSocket是类似客户端服务端的形式(采用ws协议),那么这里的WebSocketServer其实就相当于一个ws协议的Controller ;
  2. 直接@ServerEndpoint("/imserver/{userId}") 、@Component启用即可,然后在里面实现@OnOpen开启连接,@onClose关闭连接,@onMessage接收消息等方法。
  3. 新建一个ConcurrentHashMap webSocketMap 用于接收当前userId的WebSocket,方便IM之间对userId进行推送消息。单机版实现到这里就可以。
  4. 集群版 需要借助MQ或者redis发布订阅,把消息发送到第三方;多台服务器同时消费该条消息,哪台服务器上有这个WebSocket那台服务器就推送消息(因为webSocket中session是接口无法序列化)
  5. 前端url ws://172.16.15.44:8080/webSocket/参数 注意 https 需要把 ws 改为 wss

这里采用的是redis发布订阅模式

redis配置请参考 :

https://blog.csdn.net/weixin_46841515/article/details/121190753

@Component
@ServerEndpoint("/webSocket/{userno}")
@Slf4j
public class WebSocketService extends MessageListenerAdapter {
    private Session session;
    @Autowired
    private StringRedisTemplate stringRedisTemplate = SpringUtils.getBean(StringRedisTemplate.class);
    private static ConcurrentHashMap<String,ConcurrentHashMap<String,Session>> concurrentHashMap = new ConcurrentHashMap<>();

    @OnOpen
    public void  onOpen (@PathParam(value = "userno") String param, Session session){
        this.session = session;
        ConcurrentHashMap<String,Session> sessionMap = (null==concurrentHashMap.get(param))? new ConcurrentHashMap<>():concurrentHashMap.get(param);
        sessionMap.put(session.getId(),session);
        concurrentHashMap.put(param,sessionMap);
    }

    @OnClose
    public void  onClose (Session session){
        concurrentHashMap.remove(session.getId());
    }

    @OnMessage
    public void onMessage(String message,Session session){
        try {
            //前端发送的心跳数据 记录到redis中 方便删除已断开连接webSocket
            String redisKey = "webScoket_heartbeat_"+session.getId();
            stringRedisTemplate.opsForValue().set(redisKey,String.valueOf(new Date().getTime()),10, TimeUnit.SECONDS);
            String[] split = message.split("_");
            if (0<split.length){
                JSONObject jsonObject = new JSONObject();
                jsonObject.put("examId",split[0]);
                jsonObject.put("message",message);
                //发送消息到redis webScoket_message频道中(单机版可直接webSocket发送异步消息)
                stringRedisTemplate.convertAndSend("webScoket_message",jsonObject.toJSONString());
            }
        }catch (Exception e){
            e.printStackTrace();
        }
    }

    /**
     * 发送消息到redis房间webScoket_message
     * @param userAnswerMap
     * @param optionId
     */
    public void sendMessage(Map<String, ExamAnswer> userAnswerMap,Integer optionId,Integer examId){
        try {
            List<ExamAnswer> collect =  ExamAnswerUtil.getExamAnswerList(userAnswerMap);
            JSONObject jsonObject = new JSONObject();
            jsonObject.put("optionId",optionId);
            jsonObject.put("list",collect);
            String message = jsonObject.toString();
            JSONObject messageJson = new JSONObject();
            messageJson.put("examId",examId.toString());
            messageJson.put("message",message);
            //发送消息到redis webScoket_message频道中(单机版可直接webSocket发送异步消息)
            stringRedisTemplate.convertAndSend("webScoket_message",messageJson.toJSONString());
        }catch (Exception e){
            e.printStackTrace();
        }
    }

    /**
     * 监控redis房间 webScoket_message 消费到消息之后发送webSocket异步消息
     * @param msgs
     * @param pattern
     */
    @Override
    public void onMessage(Message msgs, byte[] pattern) {
        try {
            //消息内容
            byte[] body=msgs.getBody();
            //订阅房间
            String topic=new String(pattern);
            //获取存在的房间中的用户
            String result= new String(body,"utf-8");
            JSONObject js= JSON.parseObject(result);
            String examId=js.getString("examId");
            String msg=js.getString("message");
            //发送异步消息给大屏
            ConcurrentHashMap<String, Session> map = concurrentHashMap.get(examId);
            if (null != map){
                Iterator<Map.Entry<String, Session>> iterator = map.entrySet().iterator();
                while (iterator.hasNext()){
                    Session value = iterator.next().getValue();
                    boolean open = value.isOpen();
                    if (!open){
                        map.remove(value.getId());
                        continue;
                    }
                    if (null == value) continue;
                    String redisKey = "webScoket_heartbeat_"+value.getId();
                    String time = stringRedisTemplate.opsForValue().get(redisKey);
                    if (StringUtil.isBlank(time)) {
                        map.remove(value.getId());
                        continue;
                    }
                    long longTime = Long.parseLong(time);
                    long dateTime = new Date().getTime();
                    long a = dateTime - longTime;
                    if ((dateTime - longTime)>100000){
                        map.remove(value.getId());//删除已断开的连接
                    } else {
                        try {
                            //webSocket发送异步消息
                            value.getAsyncRemote().sendText(msg);
                        }catch (Exception e){
                            continue;
                        }
                    }
                }
            }
        }catch (Exception e){
            log.info("onMessage exception");
            e.printStackTrace();
        }
    }
}
  网络协议 最新文章
使用Easyswoole 搭建简单的Websoket服务
常见的数据通信方式有哪些?
Openssl 1024bit RSA算法---公私钥获取和处
HTTPS协议的密钥交换流程
《小白WEB安全入门》03. 漏洞篇
HttpRunner4.x 安装与使用
2021-07-04
手写RPC学习笔记
K8S高可用版本部署
mySQL计算IP地址范围
上一篇文章      下一篇文章      查看所有文章
加:2021-11-09 19:59:31  更:2021-11-09 20:01:46 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/4 17:47:58-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码