Springboot怎么实现WebSocket通信(二)

这篇具有很好参考价值的文章主要介绍了Springboot怎么实现WebSocket通信(二)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Springboot怎么实现WebSocket通信(二)

前言

上一篇文章分享了单机模式下,websocket的基本使用方法,但在实际的业务中,通常是不会这样使用的,大部项目都是分布式部署的,一个工程布署了多个服务节点,前端并不直接请求具体服务节点,而是先到nginx或其他代理服务器,通过nginx的负载均衡机制再转发到具体的服务节点。这种场景下,产生了几个问题:

第一,前端发起websocket连接时,是应该先连接到nginx,再由nginx转发到具体的节点?还是直接连接到具体的服务节点?

第二,nginx作为后端服务节点的代理者,nginx如何配置才能实现websocket协议的连接请求的转发?

第三,同一浏览器端发起的同一个websocket请求的session如何在多个服务节点之间实现共享?

这篇文章来和大家分享一下,实际业务开发中如何解决上述问题。

分布式场景下的websocket的工作流程

实际业务开发中,大部分项目都是采用前后端分离的开发方式和部署方式,对于http请求来说,通常是浏览器端作为客户端向服务器端发起http请求,http请求会先到nginx,经过nginx的负载均衡机制处理后,最终会把http请求转发到众多服务节点中的一个,完成业务逻辑处理后,把结果响应到客户端,客户端收到数据后渲染在浏览器中。

而对于websocket协议来说,分布式场景下的工作流程和http也比较类似,只是协议本身来说各有各的不同,前端浏览器在向服务器端发起websocket连接时,连接请求也是先到nginx上,由nginx来完转发,最终会有一个服务节点与浏览器端建立websocket连接;

浏览器端与服务器端建立websocket连接后,浏览器端和服务器端就之间操持了一个真正的长连接,浏览器端可以给服务器端发送消息,服务器端也可以给浏览器端发送消息了;

Springboot怎么实现WebSocket通信(二)

nginx与websocket

nginx从1.3版本开始是支持websocket协议的,虽然websocket协议与http协议不同,但是websocket建立连接前的握手与http是兼容的,HTTP升级机制使用Connection头Upgrade将连接从http升级到websocket;

Springboot怎么实现WebSocket通信(二)

如果nginx要实现分布式场景下websocket连接的转发,需要额外的配置,实现浏览器端和服务器端之间建立隧道,额外的配置即Upgrade的相关配置,nginx.conf的配置如下:


worker_processes  1;

events {
    worker_connections  1024;
}


http {
    include       mime.types;
    default_type  application/octet-stream;


    sendfile        on;
    
keepalive_timeout  65;

    upstream lezu{
         #模拟的服务节点1
    server 127.0.0.1:8080 weight=1;
         #模拟的服务节点2
    server 127.0.0.1:8081 weight=1;
        }

    server {
        listen       80;
        server_name  localhost;



        location / {
            root   html;
            index  index.html index.htm;
        proxy_pass http://lezu;
        # 重点的两行配置
            # 将nginx的请求头从http升级到websocket.
              proxy_set_header Upgrade $http_upgrade;
              #进行nginx连接websocket
            proxy_set_header Connection "upgrade";
            
            }

        

         error_page   500 502 503 504  /50x.html;
         location = /50x.html {
            root   html;
            }

       

       

       
    }

}

浏览器端在发起websocket连接请求时,请求的服务器端地址是nginx的代理地址,连接建立后的消息发送、接收以及连接关闭等具体的使用方法和单机模式下是一模一样的;

<!DOCTYPE HTML>
<html lang="en">
<head>
    <title>websocket测试</title>
    <meta charset="UTF-8">
    <script src="https://code.jquery.com/jquery-3.1.1.min.js"></script>
</head>
<body>
<input type="button" name="" id="open" value="建立连接"/>
<input type="button" name="" id="close" value="关闭连接">
<input type="button" name="" id="send" value="发送信息">
<div id="message"></div>
</body>
<script type="text/javascript">
    var websocket = null;
    $("#open").click(function () {
        if (websocket != null) {
            if (websocket.readyState == 1) {
                throw "连接已建立!"
            }
        }
        //判断当前浏览器是否支持WebSocket
        if ('WebSocket' in window) {
            //nginx的代理地址
            websocket = new WebSocket("ws://172.18.229.61/websocket/gaox2");
        } else {
            alert('Not support websocket')
        }
        //连接发生错误的回调方法
        websocket.onerror = function () {
            document.getElementById('message').innerHTML += ("发生错误") + '<br/>';
        };
        //连接成功建立的回调方法
        websocket.onopen = function (event) {
            document.getElementById('message').innerHTML += ("建立连接") + '<br/>';
            console.log("建立连接");
        }
        //接收到消息的回调方法
        websocket.onmessage = function (event) {
            console.log(event.data);
            document.getElementById('message').innerHTML += event.data + '<br/>';
        }
        //连接关闭的回调方法
        websocket.onclose = function () {
            document.getElementById('message').innerHTML += ("关闭连接") + '<br/>';
            console.log("关闭连接");
        }
        //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常。
        window.onbeforeunload = function () {
            alert("已关闭连接");
            websocket.close();
        }
    })
    var i = 0;
    $("#send").click(function () {
        for (let j = 0; j < 100; j++) {
            i = i + 1;
            websocket.send("hello" + i);
        }
    })
    $("#close").click(function () {
        websocket.close();
        i=0;
    })
</script>
</html>

session共享

通过nginx的关于websocket协议的相关配置后,浏览器端与服务器端一个具体的服务节点建立了websocket连接,但这里需要特别注意的是,如果服务器端主动发消息给浏览器端就会用到session,这个session是javax.websocket.Session,是不能序列化的,所以不能像http请求中把session放入到缓存中间件中来实现共享,这样就会出现下面的情况:两个浏览器端用户A和B可能会连接到两个不同的服务节点上,这个时候如果用户A想给用户B发一个消息就有问题了,用户A的websocket连接的session在服务节点1,用户B的websocket连接的session在服务节点2,session又是jvm级别的,无法实现共享,那么用户A和B之间是无法通信的。

Springboot怎么实现WebSocket通信(二)

解决这个问题的思路也很简单,实际上是没有太好的办法了,具体就是:

1、在上一篇文章里,浏览器端与服务器端成功建立websocket连接后,会把连接信息包含session缓存起来;

2、用户A的消息发送到服务器端的服务节点1时,服务节点1利用第三方的消息中间件的发布订阅模式,这里以redis为例,对其他节点广播该消息;

3、所有的服务节点都订阅这个主题的消息,收到订阅消息的服务节点,再根据消息的接收方身份标识查找接收方与服务器端的session,如果接收方与服务器端的连接没有中断,一定可以找到用户B与服务器端的websocke连接的session,找到之后就可以把消息发送给用户B了;

Springboot怎么实现WebSocket通信(二)

WebSocket业务场景

基于websocket的通信特点,大概的应用场景有:

1、在线聊天,实时消息发送;

2、视频弹幕实时更新;

3、重要通知、资讯的实时更新;

当然,实际的业务场景不止这些,有的场景需要考虑websocket session共享的问题,有的则不需要,但是在设计相关业务功能实现方案的时候,一定得考虑到websocket session共享的问题;

总结

至此,websocket的使用方法以及如何应用到实际的业务开发中的坑算是填平了。在这两篇文章的输出过程中,除了websocket本身,我get到了两点:

1、学习是一个循序渐进的过程,先学会基本使用,再深入研究更高级的用法,会比较容易且符合自然的思考逻辑;

2、任何一个相对比较新的东西,如果没有正式应用到生产的经验,则一定要根据实际业务场景反复思考、认证、测试,做出最合理的设计;

其实这篇文章还有另外一个坑未真,那就是只考虑了session共享的问题,还未考虑实际业务场景中高并发高可用情况下,websocket资源消耗与评估的问题,这个坑就靠开发和测试的同学在做压力测试的时候填上了。文章来源地址https://www.toymoban.com/news/detail-480571.html

到了这里,关于Springboot怎么实现WebSocket通信(二)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • SpringBoot集成WebSocket实现客户端与服务端通信

    话不多说,直接上代码看效果! 一、服务端: 1、引用依赖 2、添加配置文件 WebSocketConfig 3、编写WebSocket服务端接收、发送功能   声明接口代码:   实现类代码: 4、如果不需要实现客户端功能,此处可选择前端调用,奉上代码 二、客户端: 1、引用依赖 2、自定义WebSocket客

    2024年01月23日
    浏览(40)
  • SpringBoot集成WebSocket实现客户端与服务端长连接通信

    场景: 1、WebSocket协议是用于前后端长连接交互的技术,此技术多用于交互不断开的场景。特点是连接不间断、更轻量,只有在关闭浏览器窗口、或者关闭浏览器、或主动close,当前会话对象才会关闭。 2、相较于 Http/Https 通信只能由客户端主动发起请求,而 Socket 通信不仅能

    2024年02月02日
    浏览(43)
  • SpringBoot项目整合WebSocket+netty实现前后端双向通信(同时支持前端webSocket和socket协议哦)

    目录   前言 技术栈 功能展示 一、springboot项目添加netty依赖 二、netty服务端 三、netty客户端 四、测试 五、代码仓库地址   专属小彩蛋:前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站(前言 - 床长人工智能教程

    2024年02月12日
    浏览(39)
  • Spring Boot进阶(49):SpringBoot之集成WebSocket实现前后端通信 | 超级详细,建议收藏

            在上一期,我对WebSocket进行了基础及理论知识普及学习,WebSocket是一种基于TCP协议实现的全双工通信协议,使用它可以实现实时通信,不必担心HTTP协议的短连接问题。Spring Boot作为一款微服务框架,也提供了轻量级的WebSocket集成支持,本文将介绍如何在Spring Boot项

    2024年02月14日
    浏览(28)
  • Spring Boot进阶(49):实时通信不再是梦想,SpringBoot+WebSocket助你轻松实现前后端即时通讯!

            在上一期,我对WebSocket进行了基础及理论知识普及学习,WebSocket是一种基于TCP协议实现的全双工通信协议,使用它可以实现实时通信,不必担心HTTP协议的短连接问题。Spring Boot作为一款微服务框架,也提供了轻量级的WebSocket集成支持,本文将介绍如何在Spring Boot项

    2024年02月11日
    浏览(41)
  • springboot websocket通信

    目录 一、websocket是什么 二、实现websocket 2.1参考学习b站资料(一定要看,前后端详细) 2.2学习配套代码 WebSocket_ohana!的博客-CSDN博客 01-websocket协议及实现_哔哩哔哩_bilibili 1.导入pom依赖 2.编写配置类 3.所需要的pojo类:  4.需要的工具类,用于转化信息格式  5.登录的验证以及

    2024年02月08日
    浏览(24)
  • SpringBoot 集成 WebSocket 通信技术

    作者:禅与计算机程序设计艺术 WebSocket 是一种基于 TCP 的协议,它提供双向通信信道,可以实现浏览器和服务器端的全双工通信。通过 WebSocket ,网页应用可以实时地接收服务端传来的消息,并根据需要给予响应;也可以将用户操作如鼠标拖动、键盘输入等实时发送到服务端

    2024年02月07日
    浏览(32)
  • SpringBoot WebSocket配合react 使用消息通信

    引入websocket依赖 配置websocket websocket核心代码 react 前端部分代码

    2024年02月13日
    浏览(33)
  • 浅测SpringBoot环境中使用WebSocket(多端实时通信)

    目录 概述 测试:前端代码 后端代码(SpringBoot环境)  1.创建处理器类(用于处理连接和消息) 2.创建配置类(用于注册处理器类,开启WebSocket) 连接测试  这篇博客主要是记录测试使用WebSocket的过程。 需求说明 :最近在做一个预约系统,有用户端和医生端,用户端展示着

    2024年02月09日
    浏览(24)
  • SpringBoot整合Websocket(Java websocket怎么使用)

    WebSocket 是一种基于 TCP 协议的全双工通信协议,可以在浏览器和服务器之间建立 实时、双向的数据通信 。可以用于在线聊天、在线游戏、实时数据展示等场景。与传统的 HTTP 协议不同,WebSocket 可以保持 长连接 ,实时传输数据,避免了频繁的 HTTP 请求和响应,节省了网络带

    2024年02月10日
    浏览(29)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包