文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

springboot集成websocket

2023-08-16 17:15

关注

springboot集成websocket

1. 前言

​ 这里我们使用springboot搭建一个轻量级的websocket服务,同时提供4个入参。使用websocket服务可以轻松和微信小程序、支付宝小程序、网页就行双向通讯,非常实用方便。

  1. wss地址
    这里是我们自己搭建的中转服务websocket地址。

    wss://xxxx.cn/netgate/auth/pid/sn/openid
  2. 参数说明

    参数名称参数说明
    auth自定义标识符,用于鉴权处理
    pid设备产品ID【设备二维码中包含此信息】
    sn设备序列号【设备二维码中包含此信息】
    openid用户自己的id。可以使用个人平台的用户唯一id。相同ID会挤掉前一个用户的ws连接

2. 引入依赖

      org.springframework.boot     spring-boot-starter-websocket 

3. 配置文件

​ 新增WebSocketConfig.java的配置文件。用来初始化websocket

@Configuration@EnableWebSocketpublic class WebSocketConfig implements WebSocketConfigurer{    @Override    public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {       registry.addHandler(webSocketHandler(),"/netgate/{auth}/{pid}/{sn}/{openid}")                          //注册Handler           .addInterceptors(new WebSocketHandshakeInterceptor())           //注册Interceptor           .setAllowedOrigins("*");     //注册Interceptor//     //2.注册SockJS,提供SockJS支持(主要是兼容ie8)//     String sockjs_url = "/sockjs/socketServer.do";                          //设置sockjs的地址//     registry.addHandler(netgateHandler, sockjs_url)                         //注册Handler//         .addInterceptors(new WebSocketHandshakeInterceptor())               //注册Interceptor//         .withSockJS();                          //支持sockjs协议    }    @Bean    public ServletServerContainerFactoryBean createWebSocketContainer() {        ServletServerContainerFactoryBean container = new ServletServerContainerFactoryBean();        container.setMaxTextMessageBufferSize(2*1024*1024);//8192*1024 1024*1024*1024        container.setMaxBinaryMessageBufferSize(2*1024*1024);        container.setAsyncSendTimeout(55000l);        container.setMaxSessionIdleTimeout(55000l);//心跳        return container;    }    @Bean    public TextWebSocketHandler webSocketHandler() {        return new NetgateHandler();    }}

4. Websocket握手过滤器

​ 过滤器的作用主要是用来做连接接入的鉴权,和参数解析。

​ 新增WebSocketHandshakeInterceptor.java文件

public class WebSocketHandshakeInterceptor implements HandshakeInterceptor {    private final static Logger LOGGER = LoggerFactory.getLogger(WebSocketHandshakeInterceptor.class);     public boolean beforeHandshake(ServerHttpRequest request, ServerHttpResponse serverHttpResponse, WebSocketHandler webSocketHandler, Map attributes) throws Exception {        if (request instanceof ServletServerHttpRequest) {        String path = request.getURI().getPath();        if(requestIsValid(path)){        String[] params = getParams(path);        attributes.put("WEBSOCKET_AUTH", params[0]);        attributes.put("WEBSOCKET_PID", params[1]);        attributes.put("WEBSOCKET_SN", params[2]);        attributes.put("WEBSOCKET_OPENID", params[3]);        attributes.put("WEBSOCKET_FIRSTONE","yes");        }        }        System.out.println("================Before Handshake================");        return true;    }     public void afterHandshake(ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse, WebSocketHandler webSocketHandler, Exception e) {    System.out.println("================After Handshake================");    if(e!=null) e.printStackTrace();    System.out.println("================After Handshake================");    }        private boolean requestIsValid(String url){        //在这里可以写上具体的鉴权逻辑    boolean isvalid = false;    if(StringUtils.isNotEmpty(url)    && url.startsWith("/netgate/")    && url.split("/").length==6){    isvalid = true;    }    return isvalid;    }        private String[] getParams(String url){    url = url.replace("/netgate/","");    return url.split("/");    }    }

5. Websocket处理器

​ 在这里可以做消息的接收和发送。

​ 这里MqttGateway是springboot整合MQTT客户端的服务类。具体可以参考下一篇的springboot集成mqtt

​ 新建NetgateHandler.java文件

@Componentpublic class NetgateHandler extends TextWebSocketHandler {@Autowiredprivate MqttGateway mqttGateway;        private static ConcurrentHashMap> netgates = new ConcurrentHashMap>();            @Override    protected void handleTextMessage(WebSocketSession session, TextMessage message) throws Exception {    if(!session.isOpen()) {System.out.println("连接已关闭,不再处理该连接的消息!");return;}    String mes = ObjectUtils.toString(message.getPayload(),"");    String pid = session.getAttributes().get("WEBSOCKET_PID").toString();    String sn = session.getAttributes().get("WEBSOCKET_SN").toString();if(message==null || "".equals(mes)){System.out.println(getSysDate()+"============接收到空消息,不予处理。");}else if(mes.length()==1){//心跳消息过滤掉return;}else {//转发成mqtt消息String topic = "pay/"+pid+"/server/"+sn;System.out.println(topic);System.out.println(getSysDate()+"============消息处理完成:"+mes);mqttGateway.sendToMqtt(topic,mes);}    }          @Override    public void afterConnectionEstablished(WebSocketSession session) throws Exception {    System.out.println(getSysDate()+"============正在初始化连接:"+session.getId());        try {            //初始化连接,把session存储起来this.initUsers(session);} catch (Exception e) {System.out.println(getSysDate()+"============初始化连接异常-开始:"+e.getMessage());e.printStackTrace();System.out.println(getSysDate()+"============初始化连接异常-结束:"+e.getMessage());}        System.out.println(getSysDate()+"============初始化连接完成:"+session.getId());    }         @Override    public void afterConnectionClosed(WebSocketSession session, CloseStatus status) throws Exception {    System.out.println(getSysDate()+"============正在关闭连接:"+session.getId()+",isOpen:"+session.isOpen()+";code:"+status.getCode());    try {    System.out.println("断开连接状态值"+status.getCode());this.removeSession(session);} catch (Exception e) {System.out.println(getSysDate()+"============关闭连接异常-开始:"+e.getMessage());e.printStackTrace();System.out.println(getSysDate()+"============关闭连接异常-结束:"+e.getMessage());}    System.out.println(getSysDate()+"============正在关闭完成:"+session.getId()+",isOpen:"+session.isOpen()+";code:"+status.getCode());    }         @Override    public void handleTransportError(WebSocketSession session, Throwable exception) throws Exception {    System.out.println(getSysDate()+"============发生传输错误:"+session.getId()+";session.isOpen():"+session.isOpen()+";exception:"+exception.getMessage());    exception.printStackTrace();    if (session.isOpen()) {        //try { session.close(); } catch (Exception e) {e.printStackTrace();}        }else {        try {    this.removeSession(session);    } catch (Exception e) {    System.out.println(getSysDate()+"============传输错误处理异常-开始:"+e.getMessage());    e.printStackTrace();    System.out.println(getSysDate()+"============传输错误处理异常-结束:"+e.getMessage());    }        }    System.out.println(getSysDate()+"============错误处理结束:"+session.getId()+";session.isOpen():"+session.isOpen()+";exception:"+exception.getMessage());    }public synchronized void sendMsgToNetgateSn(String sn, String msg)  {if(netgates.size()>0 && netgates.containsKey(sn) && !netgates.get(sn).isEmpty()){//获取EID对应的后台管理连接 多个for (WebSocketSession ws: netgates.get(sn).values()){System.out.println("对网关指令开始发送啦:sn="+sn+"消息内容"+msg);try {ws.sendMessage(new TextMessage(msg));} catch (IOException e) {System.out.println(getSysDate()+"发生了异常:"+e.getMessage());e.printStackTrace();continue;}}}}    //连接接入的处理方法private synchronized void initUsers(WebSocketSession session){String pid = (String) session.getAttributes().get("WEBSOCKET_PID");String sn = (String) session.getAttributes().get("WEBSOCKET_SN");String openid = (String) session.getAttributes().get("WEBSOCKET_OPENID");if(StringUtils.isNotEmpty(pid) && StringUtils.isNotEmpty(sn) && StringUtils.isNotEmpty(openid)){ConcurrentHashMap netgate = netgates.get(sn);if(netgate == null){netgate = new ConcurrentHashMap();}WebSocketSession session_exist = netgate.get(sn);if(session_exist!=null) {System.out.println("检测到相同SN重复连接,SN:"+sn+",连接ID:"+session_exist.getId()+",准备清理失效的连接。。。");try {session_exist.close();} catch (IOException e) {e.printStackTrace();}}netgate.putIfAbsent(openid, session);netgates.put(sn,netgate);}}    //连接被关闭时处理集合private synchronized void removeSession(WebSocketSession session){String sn = (String) session.getAttributes().get("WEBSOCKET_SN");String openid = (String) session.getAttributes().get("WEBSOCKET_OPENID");if(netgates.get(sn).containsKey(openid)) {WebSocketSession exist_session = netgates.get(sn).get(openid);//确保是同一个session 不是同一个session则不应该进行下一步的处理if(exist_session.getId()!=null && exist_session.getId().equals(session.getId())) {netgates.get(sn).remove(openid);System.out.println("有一网关连接关闭!SN:"+sn+",当前在线数量为"+netgates.get(sn).keySet().size());}else {System.out.println("检测到关闭session异常,程序中断处理,关闭sessionId:"+session.getId()+",当前实际sessionId:"+exist_session.getId());}}else {System.out.println("检测到关闭session异常,程序中断处理,系统中未找到对应的session,Sn="+sn+"openid="+openid);}}private String getSysDate() { SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");//设置日期格式        return df.format(new Date());}}

6. 下一篇

springboot集成mqtt

来源地址:https://blog.csdn.net/qq_35921773/article/details/127451903

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯