Springboot整合Websocket遇到的坑

一、使用Springboot内嵌的tomcat启动websocket

1.添加ServerEndpointExporter配置bean

@Configuration
public class WebSocketConfig {/*** 服务器节点** 如果使用独立的servlet容器,而不是直接使用springboot的内置容器,就不要注入ServerEndpointExporter,因为它将由容器自己提供和管理* @return*/@Beanpublic ServerEndpointExporter serverEndpointExporter() {return new ServerEndpointExporter();}}

2.在接收连接的类加上@ServerEndpoint和@Component

@ServerEndpoint("/connect")
@Component

二、使用外部tomcat容器启动websocket

1.删除ServerEndpointExporter配置bean

2.接收连接的类删除@Component

三、websocket关闭连接异常

如果客户端关闭了websocket,但服务端没有监听到关闭事件,即onClose方法没有调用,这是会发生的情况

此时如果服务端向客户端推送消息,会出现异常告诉开发者:关闭了一个连接,并重新调用onClose方法

websocket 分布式开发,websocket session不支持序列化,无法存储至redis

单websocket服务器在面对并发量很大时压力会很大,而且session储存在Map中,内存压力也会很大。于是考虑分布式。

但是分布式存在websocket session共享问题,于是考虑radis存储session,但是遇到websocket session不支持序列化,无法存储。

一番搜索后有了以下几个方案

  1. 使用spring session自定义session.

  2. 既然无法序列化session,那还是存储在Map中,各服务器通过发布订阅变相实现共享websocket session.

Websocket相关问题总结(Session共享,用户多端登录等)

我们在使用websocket的时候其实主要面对的问题就是session共享的问题:

不管是基于Spring实现的Websocket的WebsocketSession

还是基于JDK实现的Session

亦或者基于netty实现的ChannelHandlerContext

用图来描述下场景吧:

OK,大家看到这个图了,差不多应该明白了Session共享应该怎么处理了。其实原理很简单:

1、我们知道nginx有IP保持的功能,其实这个功能就能解决大部分场景的Session共享问题。 但是某些极限情况下还是会有问题,比如在浏览器没有关闭的情况下同一个用户更换了网络的情况导致IP变了,或者对于某些网络的IP是变动的情况下,就会出现Session找不到的情况。

2、基于上述nginx的原理我们可以进行优化,还是单例存储。那么要操作的时候,我告诉所有的服务端,你们去找这个用户的Session,并把消息带过去。那么相应的节点根据用户拿到Session了就可以进行处理了。

上面2点大概简单的描述了下Session共享的原理,那么有这么个场景,文字可能不太好表达,我们还是用图来说明:

一般出现多端情况也应该就上面2种情况,要么允许,要么不允许。我这里简单的说下不允许的处理流程。

建立连接的时候,先获取老的Session

 Session oldSession = SOCK_MAP.get(baseStudentInfo.getId());

存在,则推送关闭消息,不存在告知其他节点去清楚。当然本节点的的Server要排除在外,这里就通过IP判断即可。

        if(oldSession!=null) {oldSession.getBasicRemote().sendObject(close);}else{//关闭其他节点的的sessionauthService.pushCloseMessage(close);}//替换SOCK_MAP.put(baseStudentInfo.getId(),session);

消息监听

            String serverIp = IPUtils.getLocalhostIp();logger.info("当前IP:"+serverIp);logger.info("content的IP:"+wsMessage.getBody().getContent());//IP不相等,说明不是当前连接的服务端,关闭其他端口if(!serverIp.equals(wsMessage.getBody().getContent())){//关闭session,并返回给前端customerHandler.closeSession(wsMessage.getBody().getReceiver(), wsMessage);}

关闭的方法:

    /*** 关闭Session* @param studentId* @param closeMessage*/public void closeSession(Long studentId,WsMessage closeMessage){Session session = SOCK_MAP.get(studentId);if(session!=null) {try {session.getBasicRemote().sendObject(closeMessage);SOCK_MAP.remove(studentId);//清除redislogger.info("连接已关闭:" + studentId);} catch (Exception e) {e.printStackTrace();logger.error("关闭连接异常");}}}

这样基本就避免多端登录的问题,如果允许多端登录的时候只需要更改存储,更改发送消息变成群发即可。

Springboot整合Websocket遇到的坑_websocket session不支持序列化,无法存储至redis_Websocket相关问题总结(Session共享,用户多端登录等)相关推荐

  1. SpringBoot 整合WebSocket 简单实战案例

    前言 这个简单实战案例主要目的是让大家了解websocket的一些简单使用. 另外使用stomp方式的: <Springboot 整合 WebSocket ,使用STOMP协议 ,前后端整合实战 ...

  2. springboot整合websocket实现消息推送

    springboot整合websocket 1.WebSocket介绍与原理 介绍:WebSocket是HTML5一种新的协议.它实现了浏览器与服务器全双工通信.一开始的握手需要借助HTTP请求完成. ...

  3. SpringBoot整合websocket实现在线客服聊天

    websocket最伟大之处在于服务器和客户端可以在给定的时间范围内的任意时刻,相互推送信息. 浏览器和服务器只需要要做一个握手的动作,在建立连接之后,服务器可以主动传送数据给客户端,客户端也可以随时 ...

  4. Springboot 整合 WebSocket ,使用STOMP协议+Redis 解决负载场景问题(二)

    前言 上一篇,简单给大家整合了一下websocket,使用stomp方式. 这篇,就是考虑到单体的服务使用websocket ,按照上一篇的整合,确实没问题. 但是如果一旦是负载多台服务的时候,那么就 ...

  5. 最简单的springboot整合websocket方式

    简介 WebSocket是一种与HTTP不同的协议.两者都位于OSI模型的应用层,并且都依赖于传输层的TCP协议. 虽然它们不同,但是RFC 6455中规定:it is designed to wor ...

  6. Springboot 整合Websocket+Stomp协议+RabbitMQ做消息代理 实例教程

    前言 如果你还没有了解过websocket,关于整合websocket的简单入门使用,可以先看看我这篇: <SpringBoot 整合WebSocket 简单实战案例> https://b ...

  7. 在线聊天室的消息单聊的实现——springboot整合WebSocket(二)

    一.声明 项目的搭建请大家移步到:在线聊天室的消息群聊的实现--springboot整合WebSocket(一) 单聊的实现是在群聊项目上进行延申改造的. 二.引入依赖 <dependency& ...

  8. Springboot整合WebSocket(基于Stomp)

    Springboot整合WebSocket(基于Stomp) 文章目录 Springboot整合WebSocket(基于Stomp) 参考链接 前言 STOMP 定义 STOMP Over WebSo ...

  9. 【Spring Web教程】SpringBoot 整合WebSocket

    概括 WebSocket是一种服务端和网页之间的通讯协议,服务端跟网页保持着长连接,可以达到服务端主动给网页发消息的功能. 常用场景:Web聊天室.通知和紧急告警.网页消息通信.物联网通讯. Spri ...

最新文章

  1. windows 下搭建Web服务器
  2. [排序算法] 选择排序(2种)
  3. Codeforces 963A Alternating Sum 【数论+数学】
  4. 2019-3-15 模拟赛 T1
  5. frame中src怎么设置成一个变量_自动格式化打印变量HMLog介绍
  6. Linux中断子系统
  7. CocoaChina2013开发者大会演讲稿-主会场-陈昊芝
  8. easyUi load方法重新加载表单的数据
  9. e5cc温控仪通讯参数设定_应用 | 如何实现S7300与S7200smart通讯?
  10. 放眼世界,一个人重整WP的整套代码,唯吾一人
  11. c语言杭电oj1090答案,杭电OJ水题答案.doc
  12. 如何清理系统大量的残余文件和系统垃圾文件?(win10)
  13. html链接外部样式表、链接网站图标
  14. 网站收录量如何提高?
  15. Linux基础入门 -用户与文件操作
  16. 记录一下tomcat的./startup.sh完成之后没有tomcat进程问题
  17. 【移动网络】Ch. 1 5G标准化与频谱
  18. 画图用计算机显卡,1分钟看懂显卡显卡绘图原理!
  19. (DDD)领域驱动设计的边界划分
  20. 对于JAVA中count=count++的理解

热门文章

  1. java快捷键禁用_pycharm 掌握这些快捷键,你就是大神!!
  2. python中的wx_配置 Python的wxWidgets可视开发环境 | 学步园
  3. java main 命令行_java Main 命令行
  4. Python程序查找表示O(1)复杂度的数字所需的位数
  5. Java ArrayList isEmpty()方法与示例
  6. Java SecurityManager checkPackageDefinition()方法与示例
  7. 2 使用_索尼黑卡RX100M6的使用指南2
  8. 数据库缓冲池_块缓冲| 数据库管理系统
  9. 如何在Java中对Collection对象进行排序?
  10. 面试官:this和super有什么区别?this能调用到父类吗?