实时消息推送系统,写得太好了!

websocket 协议是在 http 协议上的一种补充协议,是 html5 的新特性,是一种持久化的协议。其实 websocket 和 http 关系并不是很大,不过都是属于应用层的协议,接下来我们就开始实战。

websocket 定时推送

本教程基于 springboot 为脚手架,没使用过 springboot 同学可以看往期文章,或者直接去 spring 官网拉一个 springboot 基础项目下来。

加入依赖

在 springboot 的项目中添加一下 webSocket 依赖,一般一项新技术的引入在 springboot 中也只是引用一个此技术 starter 的依赖,其他配置基本 springboot 帮我们解决了。

 
  1. <dependency>

  2. <groupId>org.springframework.boot</groupId>

  3. <artifactId>spring-boot-starter-websocket</artifactId>

  4. </dependency>

配置

新建一个 Java 配置类,注入 ServerEndpointExporter 配置,如果是使用 springboot 内置的 tomcat 此配置必须,如果是使用的是外部 tomcat 容器此步骤请忽略。看 spring 源码中这样描述,使用此配置可以关闭 servlet 容器对 websocket 端点的扫描,这个暂时没有深入研究。

 
  1. @Configuration

  2. public class WebSocketConfig {

  3. @Bean

  4. public ServerEndpointExporter serverEndpointExporter() {

  5. return new ServerEndpointExporter();

  6. }

  7. }

核心代码

接下来最核心的类其实就是提供一个前后端交互的类实现消息的接收推送。

  • @ServerEndpoint(value = "/wsdemo") 前端通过此 URI 和后端交互,建立连接

  • @Component 不用说将此类交给 spring 管理

  • @OnOpen websocket 建立连接的注解,前端触发上面 URI 时会进入此注解标注的方法,和之前关于 DWR 文章中的 onpage 方法类似

  • @OnClose 顾名思义关闭连接,销毁 session

  • @OnMessage 收到前端传来的消息后执行的方法

 
  1. @ServerEndpoint(value = "/wsdemo")

  2. @Component

  3. public class MyWebSocket {

  4. private static int onlineCount = 0;

  5. private static CopyOnWriteArraySet<MyWebSocket> webSocketSet = new CopyOnWriteArraySet<MyWebSocket>();

  6. private Session session;

  7. @OnOpen

  8. public void onOpen(Session session) {

  9. this.session = session;

  10. webSocketSet.add(this);

  11. addOnlineCount();

  12. System.out.println("有新连接加入!当前在线人数为" + getOnlineCount());

  13. try {

  14. sendMessage("连接已建立成功.");

  15. } catch (Exception e) {

  16. System.out.println("IO异常");

  17. }

  18. }

  19. @OnClose

  20. public void onClose() {

  21. webSocketSet.remove(this);

  22. subOnlineCount();

  23. System.out.println("有一连接关闭!当前在线人数为" + getOnlineCount());

  24. }

  25. @OnMessage

  26. public void onMessage(String message, Session session) {

  27. System.out.println("来自客户端的消息:" + message);

  28. }

  29. @OnError

  30. public void onError(Session session, Throwable error) {

  31. System.out.println("发生错误");

  32. error.printStackTrace();

  33. }

  34. public void sendMessage(String message) throws IOException {

  35. this.session.getBasicRemote().sendText(message);

  36. }

  37. public static synchronized int getOnlineCount() {

  38. return onlineCount;

  39. }

  40. public static synchronized void addOnlineCount() {

  41. MyWebSocket.onlineCount++;

  42. }

  43. public static synchronized void subOnlineCount() {

  44. MyWebSocket.onlineCount--;

  45. }

  46. public Session getSession() {

  47. return session;

  48. }

  49. public void setSession(Session session) {

  50. this.session = session;

  51. }

  52. public static CopyOnWriteArraySet<MyWebSocket> getWebSocketSet() {

  53. return webSocketSet;

  54. }

  55. public static void setWebSocketSet(CopyOnWriteArraySet<MyWebSocket> webSocketSet) {

  56. MyWebSocket.webSocketSet = webSocketSet;

  57. }

  58. }

定时任务

使用 spring 的 Schedule 建立定时任务

  • @EnableScheduling 开启 spring 定时任务功能

  • @Scheduled(cron = "0/10 * * * * ?") 用于标识定时执行的方法,此处主要方法返回值一定是 void,没有入参。对应定时时间配置可以百度 cron 语法,根据自己的业务选择合适的周期
    在这类中,我们通过上面 MyWebSocket 提供的静态方法获取其中的 webSocketSet ,来获取所有此业务相关的所有 websocketsession,可以在定时任务中对 session 内容进行验证判断(权限验证等),进行发送消息

 
  1. @Component

  2. @EnableScheduling

  3. public class TimeTask

  4. {

  5. private static Logger logger = LoggerFactory.getLogger(TimeTask.class);

  6. @Scheduled(cron = "0/1 * * * * ?")

  7. public void test(){

  8. System.err.println("********* 定时任务执行 **************");

  9. CopyOnWriteArraySet<MyWebSocket> webSocketSet =

  10. MyWebSocket.getWebSocketSet();

  11. int i = 0 ;

  12. webSocketSet.forEach(c->{

  13. try {

  14. c.sendMessage(" 定时发送 " + new Date().toLocaleString());

  15. } catch (IOException e) {

  16. e.printStackTrace();

  17. }

  18. });

  19. System.err.println("/n 定时任务完成.......");

  20. }

  21. }

前端页面

前端页面可以参考使用,主要要更改调用的 url 为自己项目 URL

 
  1. <!DOCTYPE HTML>

  2. <html>

  3. <head>

  4. <title>My WebSocket</title>

  5. </head>

  6. <body>

  7. Welcome<br/>

  8. <input type="text" /><button onclick="send()">Send</button> <button onclick="closeWebSocket()">Close</button>

  9. <div>

  10. </div>

  11. </body>

  12. <script type="text/javascript">

  13. var websocket = null;

  14. //判断当前浏览器是否支持WebSocket ,主要此处要更换为自己的地址

  15. if('WebSocket' in window){

  16. websocket = new WebSocket("ws://localhost:8886/wsdemo");

  17. }

  18. else{

  19. alert('Not support websocket')

  20. }

  21. //连接发生错误的回调方法

  22. websocket.onerror = function(){

  23. setMessageInnerHTML("error");

  24. };

  25. //连接成功建立的回调方法

  26. websocket.onopen = function(event){

  27. setMessageInnerHTML("open");

  28. }

  29. //接收到消息的回调方法

  30. websocket.onmessage = function(event){

  31. setMessageInnerHTML(event.data);

  32. }

  33. //连接关闭的回调方法

  34. websocket.onclose = function(){

  35. setMessageInnerHTML("close");

  36. }

  37. //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常。

  38. window.onbeforeunload = function(){

  39. websocket.close();

  40. }

  41. //将消息显示在网页上

  42. function setMessageInnerHTML(innerHTML){

  43. document.getElementById('message').innerHTML += innerHTML + '<br/>';

  44. }

  45. //关闭连接

  46. function closeWebSocket(){

  47. websocket.close();

  48. }

  49. //发送消息

  50. function send(){

  51. var message = document.getElementById('text').value;

  52. websocket.send(message);

  53. }

  54. </script>

  55. </html>

效果演示

 
 

最后说一句(求关注!别白嫖!)

如果这篇文章对您有所帮助,或者有所启发的话,求一键三连:点赞、转发、在看。

关注公众号:woniuxgg,在公众号中回复:笔记  就可以获得蜗牛为你精心准备的java实战语雀笔记,回复面试、开发手册、有超赞的粉丝福利!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/42686.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

symbol数据类型以及应用场景

在js中,Symbol是一种基本数据类型,是在ECMAScript 6 (ES6) 中引入的新特性。表示独一无二 Symbol的定义 Symbol是不完整的构造函数&#xff0c;创建symbol对象时不需要new操作符,原因是通过 new 实例化的结果是一个 object 对象&#xff0c;而不是原始类型的 symbol。 var s…

STL--栈(stack)

stack 栈是一种只在一端(栈顶)进行数据插入(入栈)和删除(出栈)的数据结构,它满足后进先出(LIFO)的特性。 使用push(入栈)将数据放入stack,使用pop(出栈)将元素从容器中移除。 使用stack,必须包含头文件: #include<stack>在头文件中,class stack定义如下: namespace std…

Druid 连接池在很多方面表现出色,但在实际应用中也可能会遇到一些缺陷或问题。

Druid 连接池是阿里巴巴开源的一个功能强大的数据库连接池&#xff0c;它具有高性能、可靠性、可管理性、安全性和扩展性等特点。然而&#xff0c;尽管 Druid 连接池在很多方面表现出色&#xff0c;但在实际应用中也可能会遇到一些缺陷或问题。 1. **连接耗尽问题**&#xff1…

13 - matlab m_map地学绘图工具基础函数 - 介绍创建管理颜色映射的函数m_colmap和轮廓图绘制颜色条的函数m_contfbar

13 - matlab m_map地学绘图工具基础函数 - 介绍创建管理颜色映射的函数m_colmap和轮廓图绘制颜色条的函数m_contfbar 0. 引言1. 关于m_colmap2. 关于m_contfbar3. 结语 0. 引言 本篇介绍下m_map中用于创建和管理颜色映射函数&#xff08;m_colmap&#xff09;和 为轮廓图绘制颜…

基于深度学习的电影推荐系统

1 项目介绍 1.1 研究目的和意义 在电子商务日益繁荣的今天&#xff0c;精准预测商品销售数据成为商家提升运营效率、优化库存管理以及制定营销策略的关键。为此&#xff0c;开发了一个基于深度学习的商品销售数据预测系统&#xff0c;该系统利用Python编程语言与Django框架&a…

SQLite 命令行客户端 + Windows 批处理应用

SQLite 命令行客户端 Windows 批处理应用 下载 SQLite 客户端1. Bat 辅助脚本1. 执行SQL.bat执行 2. 导出Excel.bat执行效果 3. 导出HTML.bat执行效果 4. 清空-订单表.bat5. 订单表.bat 2. 测试 SQL1. 创建订单表.sql2. 插入订单表.sql3. 查询订单表.sql4. 清空订单表.sql5. 删…

Qt Qwt 图表库详解及使用

文章目录 Qt Qwt 图表库详解及使用一、Qwt 概述二、安装 Qwt1. 下载和编译 Qwt2. 在项目中使用 Qwt三、Qwt 的基本使用1. 创建一个简单的折线图2. 添加图例和自定义样式四、Qwt 的交互功能1. 启用缩放和平移2. 启用数据点选择五、Qwt 的高级特性1. 实时数据更新2. 多轴绘图六、…

nvm 管理多版本 node

1、下载 先不安装node 下载 nvm 1.1.10-setup.zip 解压&#xff1a;nvm&#xff1a;https://nvm.uihtm.com/ 新建nodejs/node、nodejs/nvm文件夹用于存放node版本和nvm安装路径 安装nvm&#xff1a;上述链接有安装教程 查看是否安装成功&#xff1a;重新打开cmd 输入 nvm nv…

Hyper-V克隆虚拟机教程分享!

方法1. 使用导出导入功能克隆Hyper-V虚拟机 导出和导入是Hyper-V服务器备份和克隆的一种比较有效的方法。使用此功能&#xff0c;您可以创建Hyper-V虚拟机模板&#xff0c;其中包括软件、VM CPU、RAM和其他设备的配置&#xff0c;这有助于在Hyper-V中快速部署多个虚拟机。 在…

深入理解基本数据结构:数组详解

引言 在计算机科学中&#xff0c;数据结构是存储、组织和管理数据的方式。数组作为最基础的数据结构之一&#xff0c;广泛应用于各种编程场景。在这篇博客中&#xff0c;我们将详细探讨数组的定义、特点、操作及其在不同编程语言中的实现。 什么是数组&#xff1f; 数组是一种…

原生android的内存性能提升方面的方案大致设计

一 测试目标&#xff1a; 以满足用户设备的内存性能和不杀后台为目标。 1&#xff1a;满足用户设备的内存性能是指不出现因为内存原因导致的安卓设备死机&#xff0c;卡顿等问题。 2&#xff1a;满足不杀后台是指整个设备使用时&#xff0c;不出现后台app被杀。 通常是估算如果…

构造函数注入@RequiredArgsConstructor

Api(tags "用户管理接口") RequiredArgsConstructor RestController RequestMapping("users") public class UserController {private final IUserService userService;PostMappingApiOperation("新增用户")public void saveUser(RequestBody U…

输入框输入值之后,检索表格中是否存在输入框中的值,存在就让当前文字为红色

this.searchValue为输入框的值 createKeywordHtml_content(data) { if (data undefined) { return data; } if (typeof data ! string) { data String(data) } let value data.replace(this.searchValue, <span style"color:#FF5555">$&</span>…

来一组爱胜品1133DN PRO打印机的照片

刚拆箱的机器正面照片 打开前盖正准备要安装原装耗材 下图是原装耗材&#xff0c;硒鼓型号是DR2833、碳粉盒型号是TN2833,鼓组件打印页数12000页&#xff0c;TN2833标准容量粉盒打印页数1600页/5%覆盖率&#xff0c;TN2833H大容量粉盒打印页数3000页/5%覆盖率、TN2833L超大容量…

Intel base instruction -- cvttsd2si

将截断双精度浮点数并转换为有符号整数。 4958: 8b 05 82 49 00 00 mov 0x4982(%rip),%eax # 92e0 <age_to_colorBase> // eaxgvar_92e0 495e: 48 8d 15 7b 49 00 00 lea 0x497b(%rip),%rdx # 92e0 <age_to_colorBase&g…

慢性肾脏病-MR+转录组文献

Identification of novel therapeutic targets for chronic kidney disease and kidney function by integrating multi-omics proteome with transcriptome - PMC (nih.gov) 数据和材料 Our pQTL summary data were acquired from previously published studies and can be f…

三. 根文件系统构建

根文件系统构建好以后就意味着我们已经拥有了一个完整的,可以运行的最小系统. 一. 根文件系统简介 根文件系统一般也叫做 rootfs&#xff0c;那么什么叫根文件系统&#xff1f;Linux 中的根文件系统更像是一个文件夹或者叫做目录,在这个目录里面会有很多的子目录,目录下的文件…

《妃梦千年》第二十九章:朝中波澜

第二十九章&#xff1a;朝中波澜 自从被封为护国夫人后&#xff0c;林清婉在大周的地位愈发稳固&#xff0c;她的智慧和勇气不仅赢得了皇上的信任&#xff0c;也获得了朝中众臣的敬佩。然而&#xff0c;这也引来了部分人的嫉妒和不满&#xff0c;他们开始在暗中谋划&#xff0…

AdaBoost集成学习算法理论解读以及公式为什么这么设计?

本文致力于阐述AdaBoost基本步骤涉及的每一个公式和公式为什么这么设计。 AdaBoost集成学习算法基本上遵从Boosting集成学习思想&#xff0c;通过不断迭代更新训练样本集的样本权重分布获得一组性能互补的弱学习器&#xff0c;然后通过加权投票等方式将这些弱学习器集成起来得到…

RightFont 8.7.0 Mac专业字体管理工具

RightFont 适用于 macOS 的终极字体管理器应用程序&#xff0c;提供无缝的字体管理体验。它结合了速度、直观的功能和专业的功能&#xff0c;使用户能够轻松预览、安装、组织和共享字体。 RightFont 8.7.0 Mac下载 RightFont 8.0的新增功能 RightFont 8.0 带来了全新的智能选…