Netty-SocketIo 完美替换 nodejs 的 socketio

news2025/7/21 5:10:41

背景

前段时间接到一个任务,用Java重构一个nodejs项目,其中用到了websocket的功能了,在nodejs项目中用的是socketio框架来实现websocket的功能,前端对应的也使用了socketio jar包。
一开始对socketio的用法并不是很清楚,以为前后端分离,框架没必要统一。所以后端用的websocket来实现,开发时候很顺利,联调时候问题来了,用过socketio的伙伴可能知道,socketio可以通过自定义事件名称来实现浏览器和服务器之间的数据传输,很显然,websocket没有事件这一概念。为了让前端保持原有逻辑,最好后端沿用之前的socketio功能,通过查阅资料,发现Netty-socketio可以完美解决这一问题。

代码实现

导入依赖

    <!--netty-socket io-->
    <dependency>
      <groupId>com.corundumstudio.socketio</groupId>
      <artifactId>netty-socketio</artifactId>
      <version>1.7.11</version>
    </dependency>

配置类

引入socket server,并启动。通过实现 InitializingBean 接口,重写 afterPropertiesSet(),在初始化bean的时候,开启socketio服务。


import com.corundumstudio.socketio.SocketConfig;
import com.corundumstudio.socketio.SocketIOServer;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Configuration;

import javax.annotation.Resource;
import java.util.Map;

@Configuration
@Slf4j
public class SocketIOConfig implements InitializingBean {

    @Value("${server.host}")
    private String host;

    @Resource
    private SocketIOServerHandler socketIOServerHandler;

    @Override
    public void afterPropertiesSet() {
        SocketConfig socketConfig = new SocketConfig();
        // 为了确保一个进程被关闭后,即使它还没有释放该端口,其他进程可以立刻使用该端口,而不是提示端口被占用,注意此项设置必须在socket还没有绑定到本地端口之前设置,否则会导致失效
        socketConfig.setReuseAddress(true);
        // 关闭Nagle算法,即关闭消息的ack确认
        socketConfig.setTcpNoDelay(true);
        // 如果消息发送到一半,关闭连接,-1会等到消息发送完毕再执行tcp的四次挥手。如果是0,则会直接关闭
        socketConfig.setSoLinger(-1);
        com.corundumstudio.socketio.Configuration configuration = new com.corundumstudio.socketio.Configuration();
        configuration.setSocketConfig(socketConfig);
        // host在本地测试可以设置为localhost或者本机IP,在Linux服务器跑可换成服务器IP
        configuration.setHostname(host);
        configuration.setPort(9092);
        // socket连接数大小(只监听一个端口,设置为1即可)
        configuration.setBossThreads(1);
        configuration.setWorkerThreads(100);
        // 允许自定义请求
        configuration.setAllowCustomRequests(true);
        // 协议升级超时时间(毫秒),默认10秒。HTTP握手升级为ws协议超时时间
        configuration.setUpgradeTimeout(1000000);
        // Ping消息超时时间(毫秒),默认60秒,这个时间间隔内没有接收到心跳则发送超时事件
        configuration.setPingTimeout(6000000);
        // Ping消息间隔(毫秒),默认25秒,客户端向服务器发送一条心跳消息间隔
        configuration.setPingInterval(25000);
        SocketIOServer socketIOServer = new SocketIOServer(configuration);
        // 添加事件监听器
        socketIOServer.addListeners(socketIOServerHandler);
        // 启动SocketIOServer
        socketIOServer.start();
        log.info("------- SocketIOServer start finished ------server hostIp: {}", host);
    }
}

缓存类

用来区分socket连接,我这里是根据number来区分的,也可以用userID,或者其他,具体根据业务场景来定。

import com.corundumstudio.socketio.SocketIOClient;
import org.springframework.stereotype.Component;

import java.util.concurrent.ConcurrentHashMap;

@Component
public class ClientCache {

    private static ConcurrentHashMap<String, SocketIOClient> connectMap = new ConcurrentHashMap<>();

    public void saveClient(String number, SocketIOClient socketIOClient) {
        connectMap.put(number, socketIOClient);
    }


    public SocketIOClient getClient(String number) {
        return connectMap.get(number);
    }


    public void deleteCacheByNumber(String number) {
        connectMap.remove(number);
    }


    public Boolean isContainsNumber(String number) {
        return connectMap.containsKey(number);
    }


}

Socket server和Client 交互

其实就是服务端(socket server)和客户端(socket client)之间的交互

import com.corundumstudio.socketio.SocketIOClient;
import com.corundumstudio.socketio.annotation.OnConnect;
import com.corundumstudio.socketio.annotation.OnDisconnect;
import com.corundumstudio.socketio.annotation.OnEvent;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import javax.websocket.OnError;
import javax.websocket.Session;
import java.util.UUID;
import java.util.concurrent.atomic.AtomicInteger;

@Slf4j
@Component
public class SocketIOServerHandler {
    @Autowired
    private ClientCache clientCache;

    // 记录当前在线连接数
    private static AtomicInteger ONLINE_SOCKET_CLIENT_COUNT = new AtomicInteger(0);

    /**
     * 建立连接 客户端创建socket连接的时候,调用此事件
     *
     * @param client 客户端的SocketIO
     */
    @OnConnect
    public void onConnect(SocketIOClient client) {
        UUID sessionId = client.getSessionId();
        String number = client.getHandshakeData().getSingleUrlParam("number");
        // 第一次连接的时候返回connect_event_request事件给前端,前端监听到之后,发送connect_event事件给服务端 (和原nodejs逻辑保持一致)
        clientCache.saveClient(number, client);
        ONLINE_SOCKET_CLIENT_COUNT.incrementAndGet();
        log.info("socket连接建立成功, 当前在线数为: {}, sessionId = {}, number = {}", ONLINE_SOCKET_CLIENT_COUNT, sessionId, number);
    }

    /**
     * 关闭连接  前端调用socket.disconnect()时触发改事件
     *
     * @param client 客户端的SocketIO
     */
    @OnDisconnect
    public void onDisconnect(SocketIOClient client) throws Exception {
        String number = client.getHandshakeData().getSingleUrlParam("number");
        clientCache.deleteCacheByNumber(number);
        ONLINE_SOCKET_CLIENT_COUNT.decrementAndGet();
        log.info("socket连接关闭成功, 当前在线数为: {} ==> 关闭连接信息: sessionId = {}, number = {}", ONLINE_SOCKET_CLIENT_COUNT, client.getSessionId(), number);
    }



    @OnEvent("ping")
    public void ping(SocketIOClient client, SocketMessage socketMessage) {
        String number = client.getHandshakeData().getSingleUrlParam("number");
        SocketEventVo socketEventVo = new SocketEventVo("ping");
        client.sendEvent("pong", socketEventVo);
    }


    /**
     * 自定义事件,前端socket.emit('event01', content)的时候,触发此事件
     *
     * @param client 客户端的SocketIO
     */
       @OnEvent("event01")
    public void event01(SocketIOClient client, SocketMessage socketMessage) throws Exception {
        clientCache.saveClient(socketMessage.getNumber(), client);
        if (clientCache.isContainsNumber(socketMessage.getNumber())) {
            SocketEventVo socketEventVo = new SocketEventVo("event01");
            client.sendEvent("response", socketEventVo);
        }
    }



    /**
     * 报错时触发此事件
     *
     * @param client 客户端的SocketIO
     */
    @OnError
    public void onError(Session session, Throwable error) {
        log.error("SocketIO发生错误, session id = {}, 错误信息为:{}", session.getId(), error.getMessage());
    }
}

实体类

import lombok.Data;

@Data
public class SocketMessage {
    private String number;
    private String message;
    private String req;

}

import lombok.Data;

@Data

public class SocketEventVo {
    private String status;
    private String req;
    private String desc;
    private String type;

    public SocketEventVo(String req) {
        this.status = "0";
        this.req = req;
        this.desc = "success";
        this.type = "response";
    }

    public SocketEventVo() {
    }

}

前端页面

Springboot项目,在resources包下创建templates文件夹,然后创建index.html文件。

<!DOCTYPE html>
<html>
<head>
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
    <title>SocketIO Client</title>
    <base>
    <script src="https://cdn.bootcss.com/jquery/3.3.1/jquery.min.js"></script>
    <script src="https://cdn.bootcss.com/socket.io/2.1.1/socket.io.js"></script>
    <style>
        body {
            padding: 20px;
        }
        #console {
            height: 450px;
            overflow: auto;
        }
        .connect-msg {
            color: green;
        }
        .disconnect-msg {
            color: red;
        }
    </style>
</head>

<body>

<div style="width: 700px; float: left">
    <h3>Socket Client 连接</h3>
    <div style="border: 1px;">
        <label>socketio server ip:</label>
        <input type="text" id="url" value="http://127.0.0.1:9092?number=1626" style="width: 500px;">
        <br>
        <br>
        <button id="connect" style="width: 100px;">建立连接</button>
        <button id="disconnect" style="width: 100px;">断开连接</button>
    </div>

    <hr style="height:1px;border:none;border-top:1px solid black;" />

    <h3>Socket Client发送消息</h3>
    <div style="border: 1px;">
        <label>socketEvent名称:</label><input type="text" id="socketEvent" value="event01">
        <br><br>
        <button id="send" style="width: 100px;">1626发送消息</button>

    </div>

    <hr style="height:1px;border:none;border-top:1px solid black;" />

</div>
<div style="float: left;margin-left: 50px;">
    <h3>SocketIO 连接情况</h3>
    <div id="console" class="well"></div>
</div>
</body>
<script type="text/javascript">
    var socket ;
    var errorCount = 0;
    var isConnected = false;
    var maxError = 5;

    //连接
    function connect(url) {
        // 建立socket连接
        socket = io.connect(url);
        //socket.nsp = "/socketIO";//定义命名空间
        console.log(socket)

        //监听本次连接回调函数
        socket.on('connect', function () {
            isConnected =true;
            console.log("连接成功");
            serverOutput('<span class="connect-msg"><font color="blue">'+getNowTime()+'&nbsp;</font>连接成功</span>');
            errorCount=0;
        });

        //监听断开
        socket.on('disconnect', function () {
            isConnected =false;
            console.log("连接断开");
            serverOutput('<span class="disconnect-msg"><font color="blue">'+getNowTime()+'&nbsp;</font>' + '已下线! </span>');
        });

        //监听断开错误
        socket.on('connect_error', function(data){
            serverOutput('<span class="disconnect-msg"><font color="blue">'+getNowTime()+'&nbsp;</font>;' + '连接错误-'+data+' </span>');
            errorCount++;
            if(errorCount>=maxError){
                socket.disconnect();
            }
        });
        //监听连接超时
        socket.on('connect_timeout', function(data){
            serverOutput('<span class="disconnect-msg"><font color="blue">'+getNowTime()+'&nbsp;</font>' + '连接超时-'+data+' </span>');
            errorCount++;
            if(errorCount>=maxError){
                socket.disconnect();
            }
        });
        //监听错误
        socket.on('error', function(data){
            serverOutput('<span class="disconnect-msg"><font color="blue">'+getNowTime()+'&nbsp;</font>' + '系统错误-'+data+' </span>');
            errorCount++;
            if(errorCount>=maxError){
                socket.disconnect();
            }
        });
    }

    function output(message) {
        var element = $("<div>" + " " + message + "</div>");
        $('#console').prepend(element);
    }

    function serverOutput(message) {
        var element = $("<div>" + message + "</div>");
        $('#console').prepend(element);
    }

    //连接
    $("#connect").click(function(){
        if(!isConnected){
            var url =  $("#url").val();
            connect(url);
        }else {
            serverOutput('<span class="disconnect-msg"><font color="blue">'+getNowTime()+'&nbsp;</font>' + '已经成功建立连接,不要重复建立!!! </span>');
        }
    })


    //断开连接
    $("#disconnect").click(function(){
        if(isConnected){
            socket.disconnect();
        }
    })


    //发送消息
    $("#send").click(function(){
        //自定义的事件名称
        var socketEvent =  $("#socketEvent").val();
        //发送的内容
        var enData = {
            "number": "1626",
            "message": "浏览器发送给服务端的消息",
            "req": "event01"
        };
        socket.emit(socketEvent,enData,function(data1,data2){
            console.log("ack1:"+data1);
            console.log("ack2:"+data2);
        });
    })


    function getNowTime(){
        var date=new Date();
        var year=date.getFullYear(); //获取当前年份
        var mon=date.getMonth()+1; //获取当前月份
        var da=date.getDate(); //获取当前日
        var h=date.getHours(); //获取小时
        var m=date.getMinutes(); //获取分钟
        var s=date.getSeconds(); //获取秒
        var ms=date.getMilliseconds();
        var d=document.getElementById('Date');
        var date =year+'/'+mon+'/'+da+' '+h+':'+m+':'+s+':'+ms;
        return date;
    }
</script>
</html>

前端页面不生效的话,需要在yml配置中添加如下配置

thymeleaf:
  mode: HTML
  cache: true
  prefix: classpath:/templates/
  encoding: UTF-8
  suffix: .html
  check-template-location: true
  template-resolver-order: 1

引入maven依赖

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-thymeleaf</artifactId>
  <version>2.7.12</version>
</dependency>

开始测试

首先启动socket服务端,启动之后,打开index.html文件,idea会自动提示一个浏览器标识,直接点击谷歌浏览器
在这里插入图片描述
即可打开html页面
在这里插入图片描述

接下来按 F12 看一下效果。
首先,点击【建立连接】,我们可以看到connection已经变成upgrade了,upgrade的值为websocket,说明连接建立成功
在这里插入图片描述
在这里插入图片描述

点击【发送消息】,在message的位置可以看到客户端和服务端之间的消息传输

在这里插入图片描述

每个浏览器页面对应一个sessionId,可以多开几个页面,看看后台日志打印的结果,测试一下。

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

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

相关文章

Unity Ugui 顶点颜色赋值

一、效果图 如下图&#xff1a;图片和文字的颜色都可以渐变&#xff0c;透明度也可以渐变。 原理分析&#xff1a; 不管是图片Image或是文本Text&#xff0c;它们都是网络Mesh来渲染网格是由很多三角形组成&#xff0c;那么我们根据坐标修改三角形的颜色即可实现。 工程源码…

进阶JAVA篇-如何理解作为参数使用的匿名内部类与 Arrays 类的常用API(九)

目录 目录 API 1.0 Arrays 类的说明 1.1 Arrays 类中的 toString() 静态方法 1.2 Arrays 类中的 copyOfRange(int[] original, int from, int to) 静态方法 1.3 Arrays 类中的 copyOf(int[] original, int newLength) 静态方法 1.4 Arrays 类中的 setAll(do…

论文研读|TextBack: Watermarking Text Classifiers using Backdooring

目录 论文信息文章简介研究动机研究方法水印生成水印嵌入版权验证 实验结果保真度 & 有效性消融实验 方法评估相关文献 论文信息 论文名称&#xff1a;TextBack: Watermarking Text Classifiers using Backdooring 作者&#xff1a;Nandish Chattopadhyay, et al. Nanyang…

如何选择靠谱且适合自己的IC公司?(内附各大厂薪资加班情况分析)

近期&#xff0c;有不少同学私信手里有几个offer&#xff0c;却不知道该怎么选择 &#xff1f;这着实令找不到工作的小伙伴们羡慕啊&#xff0c;今天IC修真院就来给大家分析一下如何选择靠谱且适合自己的IC公司 &#xff1f; 目前市面上可选择的芯片公司有哪些&#xff1f; 关…

SLM6500 适用于单节锂电池充电芯片 2A同步降压型鲤电池充电电路

SLM6500 是一款面向5V交流适配器的2A离子电池充电器。它是采用1.5MH2固定频率的同步降压型转换器&#xff0c;因此具有高达90%以上的充电效率&#xff0c;自身发热量极小。 SLM6500包括完整的充电终止电路、自动再充电和一个精确度达土1%的4.2V预设充电电压&#xff0c…

Linux | vim的入门手册

目录 前言 一、什么是vim 二、vim编辑器的模式 1、插入模式 &#xff08;1&#xff09;用vim打开文件 &#xff08;2&#xff09;进入插入模式 2、默认模式 &#xff08;1&#xff09;光标移动 &#xff08;2&#xff09;复制、粘贴与剪切操作 &#xff08;3&#x…

毫米波雷达与其他传感器的协同工作:传感器融合的未来

随着科技的不断进步&#xff0c;传感技术在各个领域的应用愈发广泛。毫米波雷达作为一种重要的传感器技术&#xff0c;以其高精度、强穿透力和适应性强等优点&#xff0c;在军事、医疗、汽车、工业等领域都得到了广泛应用。然而&#xff0c;单一传感器的局限性也逐渐显现&#…

017 基于Spring Boot的食堂管理系统

基于Spring Boot的食堂管理系统 项目介绍 本项目是基于Java的管理系统。采用前后端分离开发。前端基于bootstrap框架实现&#xff0c;后端使用Java语言开发&#xff0c;技术栈包括但不限于SpringBoot、MyBatis、MySQL、Maven等&#xff0c;开发工具为IDEA。 功能介绍 主页 …

day28--JS(同步异步代码,回调函数地狱,promise链式调用,async函数和await,事件循环,宏任务与微任务)

目录 同步异步代码&#xff1a; 回调函数地狱&#xff1a; Promise Promise.all静态方法 链式调用 async函数和await&#xff1a; 语法&#xff1a; 捕获错误try...catch&#xff1a; 事件循环--执行过程&#xff1a; 宏任务与微任务&#xff1a; 同步异步代码&#…

OpenCV4 :并行计算cv::parallel_for_

OpenCV4 &#xff1a;并行计算cv::parallel_for_ 在计算机视觉和图像处理领域&#xff0c;OpenCV&#xff08;开源计算机视觉库&#xff09;是一个非常强大和广泛使用的库。随着图像分辨率的提高和计算任务的复杂度增加&#xff0c;实时处理变得越来越困难。为了解决这个问题&…

基于springboot实现汉服文化分享平台项目【项目源码+论文说明】计算机毕业设计

摘要 本论文主要论述了如何使用JAVA语言开发一个汉服文化平台网站 &#xff0c;本系统将严格按照软件开发流程进行各个阶段的工作&#xff0c;采用B/S架构&#xff0c;面向对象编程思想进行项目开发。在引言中&#xff0c;作者将论述汉服文化平台网站的当前背景以及系统开发的…

selenium教程 —— css定位

说明&#xff1a;本篇博客基于selenium 4.1.0 selenium-css定位 element_css driver.find_element(By.CSS_SELECTOR, css表达式) 复制代码 css定位说明 selenium中的css定位&#xff0c;实际是通过css选择器来定位到具体元素&#xff0c;css选择器来自于css语法 css定位优点…

使用cpolar内网端口映射技术实现U8用友ERP本地部署的异地访问

文章目录 前言1. 服务器本机安装U8并调试设置2. 用友U8借助cpolar实现企业远程办公2.1 在被控端电脑上&#xff0c;点击开始菜单栏&#xff0c;打开设置——系统2.2 找到远程桌面2.3 启用远程桌面 3. 安装cpolar内网穿透3.1 注册cpolar账号3.2 下载cpolar客户端 4. 获取远程桌面…

Linux 如何进行内存分配

虚拟内存管理回顾 在 Linux 操作系统中&#xff0c;虚拟地址空间的内部又被分为内核空间和用户空间两部分&#xff0c;不同位数的系统&#xff0c;地址空间的范围也不同。比如最常见的 32 位和 64 位系统&#xff0c;如下所示&#xff1a; 通过这里可以看出&#xff1a; 32 位…

【网络安全 --- MySQL数据库】网络安全MySQL数据库应该掌握的知识,还不收藏开始学习。

四&#xff0c;MySQL 4.1 mysql安装 #centos7默认安装的是MariaDB-5.5.68或者65&#xff0c; #查看版本的指令&#xff1a;[rootweb01 bbs]# rpm -qa| grep mariadb #安装mariadb的最新版&#xff0c;只是更新了软件版本&#xff0c;不会删除之前原有的数据。 #修改yum源的配…

完全掌握Nginx的终极指南:这篇文章让你对Nginx洞悉透彻

Nginx是一款轻量级的Web服务器、反向代理服务器&#xff0c;由于它的内存占用少&#xff08;一个worker进程只占用10-12M内存&#xff09;&#xff0c;启动极快&#xff0c;高并发能力强&#xff0c;在互联网项目中广泛应用。 上图基本上说明了当下流行的技术架构&#xff0c;其…

软考系统架构设计师考试冲刺攻略

系统架构冲刺攻略 上篇为综合知识&#xff0c;介绍了系统架构设计师应熟练掌握的基本知识&#xff0c;主要包括绪论、计算机系统、信息系统、信息安全技术、软件工程、数据库设计、系统架构设计、系统质量属性与架构评估、软件可靠性、软件架构的演化和维护、未来信息综合技术等…

贪心算法:猫粮兑换最大数量的五香豆

小老鼠存了一些猫粮&#xff0c;他想到猫猫库房兑换最大数量的五香豆。 (本笔记适合熟悉循环和列表的 coder 翻阅) 【学习的细节是欢悦的历程】 Python 官网&#xff1a;https://www.python.org/ Free&#xff1a;大咖免费“圣经”教程《 python 完全自学教程》&#xff0c;不…

凉鞋的 Unity 笔记 201. 第三轮循环:引入变量

201. 第三轮循环&#xff1a;引入变量 在这一篇&#xff0c;我们进行第三轮 编辑-测试 循环。 在之前我们编写了 输出 Hello Unity 的脚本&#xff0c;如下: using System.Collections; using System.Collections.Generic; using UnityEngine;public class FirstGameObject …

低代码加速软件开发进程

IT 团队依靠笨重的软件开发流程和密集型的手工编码来构建可靠的现代应用程序的时代即将结束。随着新自动化技术的兴起、开发人员的短缺&#xff0c;以及渴望创新的客户和最终用户的需求迅速提高&#xff0c;软件行业被迫寻求替代方法&#xff0c;要求不仅提供服务和产品&#x…